Show Posts

This section allows you to view all posts made by this member. Note that you can only see posts made in areas you currently have access to.


Messages - tomxp411

Pages: [1] 2
1
QB64 Discussion / Re: What's your philosophy about constants?
« on: March 24, 2022, 10:13:41 pm »
const is better for constants, you have be careful with #define/macros
suppose that your define was #DEFINE p2 Pi + Pi then x=p2*4 would become Pi+Pi*4 unless you enclose it in parentheses

He was literally describing a #define, but calling it "EQN". Hence my reply: call it DEFINE or MACRO.

2
QB64 Discussion / Re: What's your philosophy about constants?
« on: March 24, 2022, 09:37:28 pm »
Just wanted to say thanks for all comments.  I've decided to introduce a new statement called "EQN" (for equation).  Although EQN and CONST use the same background stuff to implement them, I like the semantic differentiation.  I'm not liking a CONST statement allowing appearance of not looking like a constant.

If anybody knows of any BASIC that already has an EQN statement or function, please let me know.

Imagine an income tax program or something like that, here's how EQN works:

Code: QB64: [Select]
  1. dim line1, line2, line4
  2. eqn line3 = line1 + line2 , line5 = line3 + line4
  3. line1 = 3 : line2 = 7 : line4 = 4
  4. print line1
  5. print line2
  6. print line3
  7. print line4
  8. print line5

The more correct term for this would be DEFINE, as it's already used in C/C++ for in-line macros.

#DEFINE p2 Pi * 2

x = p2 * 4;

In fact, older C compilers don't really support constants. As a result, constants in C are typically implemented via #DEFINE preprocessor directives... so are really macros. So you might think about adding your new keyword similarly.

3
QB64 Discussion / Re: What's your philosophy about constants?
« on: March 24, 2022, 02:04:10 am »
The whole point of a constant is that it can't be modified. The other point is that it's a literal replacement made at compile time.

Let's look at one of list of constants built in to the DotNet SDK, the keyboard scan codes. There are 773 keyboard scan codes in the System.Windows.Forms.Keys enumeration - just for a list of keys on the computer.

Now consider all of the other thousands of enumerations and constants in the system... the "Magic Number Database" has over 380,000. That's not all of them, either. So if you were to include all of the constants your program might need as variables every time you compile it, you would be adding in over 1.5 megabytes just for the variable storage, not to mention the other things that go into storing and retrieving a variable.

So there are two very important reasons to use CONST in your program:
1. It's an obvious semantic flag that says "this value isn't going to change."
2. Those values get converted directly to data during compile time. There's no variable storage or lookup for those values, and unused constants actually disappear from the object code, saving even more space.

Does it matter in a little dinky 30K demo? No. But does it matter in large applications, spanning thousands of source files and multiple megabytes of object code?

Absolutely.


4
Right, this is clearly not a good candidate for a recursive algorithm. Recursion is ideal for walking any sort of tree structure, since you can walk down each branch without losing track of where you started. When dealing with gaming, it's also ideal for pathfinding algorithms. Dijkstra's algorithm, for example, is the fundamental algorithm used in graph theory, and it's all about recursion. Each node is tested recursively, and the recursion terminates when a path reaches a terminal point (the goal or a dead end).

However, the classic example for recursion is reading a directory tree:

(this is psuedocode, not VB):

sub GetDirectory(path)
    Read Directory into directory()
    for each entry in directory()
        print entry
        if entry is directory then GetDirectory(entry)
    next
end sub





5
QB64 Discussion / Re: Has $Debug been working well for you?
« on: March 10, 2022, 02:54:14 am »
@tomxp411 I could not reproduce the issue regarding breakpoints, so I'd be glad if you could list the exact steps to reproduce it (provide some simple code if you can).

I can't reproduce it on a simple test program, either. I'll try it again the next time I'm working on my Trek program.

6
QB64 Discussion / Re: Has $Debug been working well for you?
« on: March 10, 2022, 02:52:15 am »
Latest dev build available now contains the new ability to choose number format, as per @RhoSigma's suggestion (dec/hex/oct/bin) as well as a fix for the issue reported by @tomxp411 regarding occasional crashes due to removing variables between $debug sessions.

@tomxp411 I could not reproduce the issue regarding breakpoints, so I'd be glad if you could list the exact steps to reproduce it (provide some simple code if you can).

Thanks.

I'll see what I can do on the stuck BP issue. I seem to recall it being kind of random, and toggling the BP the second time always fixed it.

7
Programs / Re: A Replacement Method for Data Statements
« on: March 09, 2022, 03:37:00 pm »
Thanks. That helps.

I am not sure if I've actually used QB64 on my Pi. I have been meaning to try it, as I want my programs to work on Linux, as well as Windows. I'll have to remember to start testing on the Pi, once in a while.

8
QB64 Discussion / Re: Has $Debug been working well for you?
« on: March 09, 2022, 02:53:23 pm »
I really like having the debug/watch feature, and it's helped me quite a bit.

However, there are some issues that could make it more useful:

1. If a variable being watched has been removed since the last debugging session, the debugger will crash when you hit a breakpoint.
2. Remove a BP while the program is paused on that BP... and the program will still pause there on the next cycle. I seem to remember this happening frequently.
3. It is not possible to add watch variables defined in include files. We need a way to add those variables - possibly through a different watch dialog that simply asks for the variable name and its scope.
4. With a large program, the list of watch variables can get huge... it would be nice if I could just add a watch by selecting it in the source code or typing its name, rather than having to parse that huge watch list.

In fact, the variable list is the part of the debugger I don't like. While it looks great, it's unwieldly when there are more than a few variables in the program. I'd much rather just be able to type in the scope and/or variable name:


+-------------------------------------+
|             Add Watch               |
| Sub/Function: _____________         |
| Variable:     _____________         |
| Member:       _____________         |
|    [Add Watch]        [Cancel]      |
+-------------------------------------+


9
Programs / Re: A Replacement Method for Data Statements
« on: March 09, 2022, 02:33:17 pm »
Be really cool getting QB64 working on those devices.

I checked out a Pi a couple years ago, the screen was smaller than a phone's, not going to work with old eyes!, I returned the kit and appreciate my laptop all the more!

But that kind of thing is the base of young computer hobbyists for today!

You probably got one of the handheld portable kits. The Pi is designed to use any HDMI display, so you can use it on any modern TV or computer monitor.

10
Programs / Re: A Replacement Method for Data Statements
« on: March 09, 2022, 02:26:07 pm »
No bug really, Data statements work everywhere except where you tried it in a Pi apparently.

I actually found a but in the READ statement that was fixed within an hour of me reporting it. @MrGW454 if you are not running the latest version, you'll want to compile from source. If you have a packaged binary, you may still have the bug, since there has not been an official release since then. Meaning you would still have the bug where READ sometimes grabs the last line of source code, rather than the correct DATA value.

Also... could you post the original Discord message, so we know what the actual problem is?

11
QB64 Discussion / Re: BEEP makes program load faster
« on: February 17, 2022, 12:38:10 pm »
I just tried it, and BEEP doesn't seem to actually make a difference. I tried it with my 900 line Trek program, and it takes about 5 seconds to launch (after taking a bit longer the first try that day. I blame a newly booted computer and freshly downloaded source files.)

I placed a BEEP on the first line of code that actually gets executed. It did not change the compile or launch time. Still about 5 seconds.
I took the BEEP back out. Still about 5 seconds.

So I tried another program: a simple 15 line keyboard test utility. Same deal. With or without BEEP, it takes about 4-5 seconds to compile and launch.

How big is your program? Do you have any includes? I'm thinking you are seeing the effects of caching or pre-compilation or some other optimization that takes effect after the first use. BEEP itself does not actually seem to make a difference to me.

12
QB64 Discussion / Re: Just a Code Sample: EXIT SUB
« on: February 14, 2022, 01:54:33 pm »
FWIW, while I agree that having to declare all subs at the top was a bloody nuisance, "Call subfoo" is, IMO, way more readable, than just naming the sub. Especially when others need to read and understand your code.

Not really. While it can be used as a signal that the following symbol is a subroutine, it's mostly a waste of space. Its only syntactic purpose is to allow a function-style syntax when a value is not being returned. ie: CALL OpenFile(filename) as opposed to OpenFile filename.

More to the point: a SUB is really a new command. From the developer's standpoint, you're literally adding a new command to the language when adding a subroutine. In fact, most languages rely on this and don't have any internal commands, other than basic flow control. In C, for example, there's not even a PRINT statement. Instead, the printf() function is a library routine and not actually part of the language itself.

If it helps you in your own work, there's nothing particularly wrong with using CALL, but if you're working with other people, you'll probably find that it's not recommended. Microsoft still includes the CALL statement, but their documentation now says to use it only in specific circumstances, such as when an identifier starts with a character that is not a letter.




13
QB64 Discussion / Re: True and False
« on: February 10, 2022, 03:16:19 pm »
Ok, we need to hook up at a bar and have a beer and talk this through.... so you guys are saying the IF is in fact a decision and not a simple ON arriving at the index zero, print the index zero?? The IF statement in your example Steve simply says "IF a", which I have always read "On reaching a " or " If the loop index has reached a .. print it's value"

It seems, even if I make a = 0, which should be a True statement   (ie a = 0  : If a then Print "True" else Print "False" comes up with False)   we don't take away the decisional aspect of "If a"

I appreciate the lesson here. Thanks guys.

You're over-complicating the IF statement.

The IF statement tests for zero. That's all it does. Any other comparisons are actually done by the expression in the IF statement, not IF itself.

IF <value> THEN
<value is not zero>
ELSE
<value is zero>
END IF

IF does not test for equality, greater, lesser, or anything else. It simply checks for zero. In fact, the machine code for this ends up as "Jump if Zero" (JZ) and "Jump if Not Zero" (JNZ).

So everything between IF and THEN is actually an expression., which returns a numeric result.

In BASIC, an expression is basically anything that can be turned into a value: "3" is an expression, "4  + 5" is an expression, and "A < Y" is an expression that returns a Boolean result.

And since a Boolean result is either On or Off, we treat Off as zero and On as -1 (Because signed binary integers are negative when the left bit is on. It's more complicated than that, but that's how the "negative" flag works in machine language.)

So things like IF X=Y are actually two separate operations: the expression (X=Y) is evaluated first and returned as a single value. The IF statement then jumps to the ELSE or END IF when the value is non-zero.

To understand Boolean expressions a bit more, consider these statements:

PRINT 0
PRINT 1
PRINT 0=0
PRINT 0=1
PRINT 1=1

PRINT 0=0 prints -1, because when the = is used in an expression, it's a test for equality. And a test for equality returns -1 when the two values are equal.
Likewise 0=1 returns 0, and 1=1 returns -1

Once you understand the separation between statement and expression, you'll better understand how to use the IF statement to write complex and useful tests.

14
QB64 Discussion / Re: A million dollar accountant's headache
« on: February 08, 2022, 04:25:58 pm »
All this really highlights how terrible IEEE-754 floats actually are for representing decimal values. While they're perfectly fine for values that don't need to be printed, but any time you're dealing with money or with explicit powers of 10 (ie decimal positions), it's better to use integer types and deal with the decimal yourself.

When I worked in the banking industry, we always stored values as integers. We had a special "print money" function to convert integers to dollars and cents, then print that. It went something like this:

Code: QB64: [Select]
  1. DEFLNG A-Z
  2.  
  3. PrintDec 1
  4. PrintDec 10
  5. PrintDec 100
  6. PrintDec 1000
  7. PrintDec 10000
  8. PrintDec 100000
  9. PrintDec 1000000
  10. PrintDec 10000000
  11. PrintDec 100000000
  12. PrintDec -1
  13. PrintDec -10
  14. PrintDec -100
  15. PrintDec -10000000
  16. PrintDec 12350
  17. PrintDec -12350
  18.  
  19. SUB PrintDec (N AS LONG)
  20.     M$ = _TRIM$(STR$(ABS(N)))
  21.     IF LEN(M$) < 2 THEN M$ = "00" + M$
  22.     IF LEN(M$) < 3 THEN M$ = "0" + M$
  23.  
  24.     lm = LEN(M$)
  25.     N$ = " "
  26.     IF N < 0 THEN N$ = "-"
  27.     PRINT SPACE$(10 - lm); N$; LEFT$(M$, lm - 2); "."; RIGHT$(M$, 2)
  28.  




15
QB64 Discussion / Re: Possible issue with Select Case
« on: February 07, 2022, 09:38:00 pm »
Looks like your right, though why it cares?
perhaps I have gotten to used to be able to go high to low in things like LINE and _PUTIMAGE?
and it doesn't care if I CASE 16,15,14,13,12,11.
but what ever, found why there was an issue and resolved,
Thanks Steve.

Think about how the TO operator works.  CASE X TO Y is shorthand for
IF Value >= X AND Value <= Y THEN ...

So if you plug your numbers into the test, you get:
IF 15 >= 16 AND 15 <= 11 THEN
What is the result of this test?

The reason it works with a list (16,15,14, etc) is that the value is compared separately against each item in the list. Expressed as an IF, that's:
IF 15 = 16 OR 15 = 15 OR 15 = 14 ...

Does that make sense?

Pages: [1] 2