QB64.org Forum

Active Forums => QB64 Discussion => Topic started by: Dimster on October 27, 2021, 09:01:51 am

Title: Recursion
Post by: Dimster on October 27, 2021, 09:01:51 am
I was wondering if I could get your advice on where best to apply recursive code. Now that Function has a recursive element to it, would it be to my speed advantage to explore it? I presently have in my program multiple calls to the same Subroutine. Basically the program inputs data, calls a subroutine to work with the data then onto the next piece of data. I was thinking perhaps the program may have speed advantage if it inputted and stored the data, then, once I have it all,  use a recursive function call to perform the same work on each piece of data which the Subroutine had been performing. This is the trouble with new toys...your imagination goes wild and reality is always so painfully down the road.
Title: Re: Recursion
Post by: FellippeHeitor on October 27, 2021, 09:33:44 am
Not answering your question, but just to clarify: recursion is not new in QB64. One specific scenario of functions without parameters failing to call themselves has been fixed in v2.
Title: Re: Recursion
Post by: FellippeHeitor on October 27, 2021, 09:35:18 am
And some examples of recursive functions for you to be inspired: http://www.qb64.org/wiki/STATIC
Title: Re: Recursion
Post by: Dimster on October 27, 2021, 09:58:25 am
Thanks Fell ... yes, I was aware V2.0 fixed that specific instance in Functions. I have always looked to Subroutines to do the heavy lifting and Functions to do quick math or handling Scientific Notation. When you were explaining the fix to Functions it occurred to me that Functions could do a lot more of the heavy lifting but why do it if there isn't some kind of an advantage....and thanks a lot for the "Static" . I must admit that really does give my perception of how functional functions can be.
Title: Re: Recursion
Post by: bplus on October 27, 2021, 12:15:31 pm
Here's a graphics example of recursion:
Code: QB64: [Select]
  1. _Title "Recursion example" 'b+ 2021-10-27
  2.  
  3. 'Recursion is a sub or function that calls itself until it's job is done
  4. 'Here is a grahics
  5. Const pi = _Pi
  6. Screen _NewImage(700, 700, 32)
  7.  
  8. recThis 700 / 2, 700 / 2, 350 / 2, 1
  9.  
  10. Sub recThis (x, y, arm, level As Integer)
  11.     'first thing to ask in recursive subroutine is are we done!!!
  12.     If arm < 2 Then Exit Sub ' recursion is finished
  13.     ' no not done
  14.     If level Mod 2 Then
  15.         x1 = x + arm * Cos(0): y1 = y + arm * Sin(0)
  16.         x2 = x + arm * Cos(pi): y2 = y + arm * Sin(pi)
  17.         Line (x1, y1)-(x2, y2)
  18.     Else
  19.         x1 = x + arm * Cos(pi / 2): y1 = y + arm * Sin(pi / 2)
  20.         x2 = x + arm * Cos(1.5 * pi): y2 = y + arm * Sin(1.5 * pi)
  21.         Line (x1, y1)-(x2, y2)
  22.     End If
  23.     recThis x1, y1, arm * .7, level + 1
  24.     recThis x2, y2, arm * .7, level + 1
  25.  
Title: Re: Recursion
Post by: bplus on October 27, 2021, 12:44:54 pm
I think this is an example of recursion making things more complicated to reverse some text:
Code: QB64: [Select]
  1. _Title "Recursion example" 'b+ 2021-10-27
  2.  
  3. 'Recursion is a sub or function that calls itself until it;s job is done
  4. 'Here is a text example
  5. Const pi = _Pi
  6. Screen _NewImage(700, 700, 32)
  7. s$ = "This is an example of bplus using recursion to reverse this text."
  8. Print reverseMeSomeMore$(s$, 0)
  9.  
  10. Function reverseMeSomeMore$ (text$, level)
  11.     ' are we there yet?
  12.     If level = Int(Len(text$) / 2) + 1 Then reverseMeSomeMore$ = text$: Exit Function
  13.     s$ = Mid$(text$, level, 1)
  14.     c$ = Mid$(text$, Len(text$) - level + 1, 1)
  15.     Mid$(text$, level, 1) = c$
  16.     Mid$(text$, Len(text$) - level + 1, 1) = s$
  17.     reverseMeSomeMore$ = reverseMeSomeMore$(text$, level + 1)
  18.  
  19.  
Title: Re: Recursion
Post by: Dimster on October 27, 2021, 01:27:41 pm
So do you have preference? They are short examples but I felt the Subroutine ran a little faster than the Function..then again the function had to deal with strings
Title: Re: Recursion
Post by: Dimster on October 27, 2021, 01:40:40 pm
Thanks for the examples by the way. I am loosely defining Recursion as just multiple calls. So I've looking at this from the perspective of calling the same subroutine from the main program. You've some very good example of recursive calls within the Subroutine and Function. I have never really approached recursion within the routine but that recursion within the Function is what I have been thinking of doing.

Also would appear, in terms of speed.... I should be considering the time involved in multiple calls from the main program to the same subroutine  v's recursive calls within a Function. (not exactly sure how to re-code the program to do this) But I'm thinking one call to the Function may in fact be faster.
Title: Re: Recursion
Post by: bplus on October 27, 2021, 01:49:13 pm
So do you have preference? They are short examples but I felt the Subroutine ran a little faster than the Function..then again the function had to deal with strings

It really depends on what you are trying to do.
I can't imagine drawing some things without recursion eg, trees, who wants to have to keep track of all the branching?

I've also used it for Sudoku solutions and MineSweeper for sweeping empty cells and for Paint3 that paints one color when it finds another color. Qsort uses recursion.

The Classic example of recursion is factorial but I would just use a loop for that. Recursion is cool when it splits one call to 2 or 4 for 2D work like first example above.

Oh you @Dimster posted while I was answering previous.
Title: Re: Recursion
Post by: bplus on October 27, 2021, 01:56:02 pm
Thanks for the examples by the way. I am loosely defining Recursion as just multiple calls. So I've looking at this from the perspective of calling the same subroutine from the main program. You've some very good example of recursive calls within the Subroutine and Function. I have never really approached recursion within the routine but that recursion within the Function is what I have been thinking of doing.

Also would appear, in terms of speed.... I should be considering the time involved in multiple calls from the main program to the same subroutine  v's recursive calls within a Function. (not exactly sure how to re-code the program to do this) But I'm thinking one call to the Function may in fact be faster.

The very definition of a recursive function or sub is that it calls itself to do it's job. It must call itself at least once to be considered recursive. It's like a code echo. :) Hey that gives me an idea!

Title: Re: Recursion
Post by: Dimster on October 31, 2021, 10:26:40 am
Wondering if I have this straight .. There is a danger with a Recursive procedure to cause a Stack Overflow. The solution is to use Static variables. I see where the recursive Quick Sort does not use Static variables (at least I have never declared Static variables when I use the Quick Sort method). So I'm thinking, if I create a Recursive procedure whereby a math formula is deployed (+,-,*,/) then this is were the danger lays. If the Recursion is simple comparisons, or True/False, or re-ordering then there is no threat to Overflow the Stack. Is that correct?
Title: Re: Recursion
Post by: bplus on October 31, 2021, 11:30:10 am
One thing that will save resources in a recursive QuickSort is to share the array and only work on that one specific array with the QuickSort routine, in that way it does not have to be an argument in the call to the sub and thus repeatedly called down the recursive line. In my Interpreter with Sorting, I think I copied arrays to standard shared array worked on by QuickSort routines. I even had different routines for ascending and descending so that would not have to be included in parameter list of the call to the specialized QuickSort routine.

What might surely blow the stack are recursive routines that have allot of parameters and branch out over seemingly endless lines of recursion.

Quote
The solution is to use Static variables.
Don't know about that, my instinct would would be to Share the variable and give it a name not likely to be used elsewhere.
Title: Re: Recursion
Post by: luke on October 31, 2021, 11:37:15 am
Wondering if I have this straight .. There is a danger with a Recursive procedure to cause a Stack Overflow. The solution is to use Static variables. I see where the recursive Quick Sort does not use Static variables (at least I have never declared Static variables when I use the Quick Sort method). So I'm thinking, if I create a Recursive procedure whereby a math formula is deployed (+,-,*,/) then this is were the danger lays. If the Recursion is simple comparisons, or True/False, or re-ordering then there is no threat to Overflow the Stack. Is that correct?
I think it's worth explaining exactly what a stack overflow is and why it occurs:

Every time the CPU jumps to a subroutine (a SUB or a FUNCTION), it needs to remember where in the code it came from, so it can go back there when it's finished.

The way it does this is by pushing its current address to the stack before jumping. But really, you can think of this as just having a big array in memory, and every time you need to do a subroutine you REDIM _PRESERVE the array to increase the size by one, add your current code position to the new element, then go to the subroutine. After you finish and return to where you came from, you can REDIM the array again to make it one element smaller (since you don't need the value any more).

Now, because of Reasons™ the program can only make this array so big, about 8MB in the best of cases (it could be quite a bit less). This means you can only have so many subroutines-that-call-subroutine, because eventually you'll run out of space to store these return locations. Note that this isn't a limit on the total number of subroutine calls we can make in the program in total, because whenever a subroutine returns it frees up an element of our array. It's just a limit on how many calls you can have chained together.

For further Reasons™ the data structure used to store these return locations is in fact a stack, and if it gets full it goes beyond its bounds, hence a "stack overflow" (in the real world the stack also grows downwards so the visual metaphor is a little messed up, but ignore that).

You can see now why recursion can be problematic - it's a very quick way to make many subroutine calls and fill up the stack with return locations. It doesn't matter what you're doing in the subroutine, if you're recursing you just have to make sure you have a limit on how deep the recursion goes.

Now, it turns out the exact number of recursive calls you can make is somewhat dependent on the contents of the routine: for even more Reasons™ every variable in a subroutine (that isn't SHARED, STATIC or a parameter) also takes up an element in the same stack that stores our return locations, and does so on every call. But don't be misled - even a subroutine with no local variables can cause a stack overflow, you just might have to wait a microsecond longer for it to happen.

It can be shown (by some high-pants maths) that every recursive program can actually be written as a nonrecursive one by using one or more loops. Rewriting it to remove the recursion is generally the way to go if you're having stack overflows (and it's not just a bug in your algorithm).
Title: Re: Recursion
Post by: Dimster on October 31, 2021, 12:06:17 pm
So can a Stack Overflow be trapped or monitored?
Title: Re: Recursion
Post by: bplus on October 31, 2021, 10:02:42 pm
Just like any error I suppose.
Title: Re: Recursion
Post by: TempodiBasic on November 01, 2021, 08:41:17 am
Hi boys
IMHO
the amount of available resources and the amount of data to be processed are the compass to choose between recursive function and non-recursive algorithm.
Great resources and / or little amount of data are a good index for recursive function and vice versa.
Usually I have found recursion method used for binary search ,  algorythms of ordering (sorting data),  parsing algorythm (interpreter/compiler/syntax checker), translation algorythm, ripetitive calculation (factorial numbers, math sequence like Fibonacci, Collatz conjecture).
just my two cents

Happy coding
Title: Re: Recursion
Post by: SMcNeill on November 01, 2021, 09:09:38 am
If you want the easiest example of recursion, then I'd like to add in a simple Fill routine for folks to study:

Code: QB64: [Select]
  1. Screen _NewImage(640, 480, 32)
  2.  
  3. Circle (240, 240), 200, &HFFFF0000
  4. Fill 240, 240, &HFFFF0000
  5.  
  6. Sub Fill (x, y, kolor As _Unsigned Long)
  7.     If Point(x, y) <> kolor Then Pset (x, y), kolor
  8.     If Point(x + 1, y) <> kolor Then Fill x + 1, y, kolor
  9.     If Point(x - 1, y) <> kolor Then Fill x - 1, y, kolor
  10.     If Point(x, y + 1) <> kolor Then Fill x, y + 1, kolor
  11.     If Point(x, y - 1) <> kolor Then Fill x, y - 1, kolor

Give it a center point.  It then paints the pixels around it, if they're not already the proper color.  Then it moves to those pixels and repeats the process.  On and on, until the region to be painted is filled in completely.

Note, this should probably have a simple IF x < 0 or y < 0 or X >= _WIDTH or y >=_HEIGHT Then EXIT SUB in it just to make certain that it doesn't try to work itself to death off-screen on us.  ;)
Title: Re: Recursion
Post by: Dimster on November 01, 2021, 10:01:19 am
@bplus and @luke  ... Trapping I guess would be error code 256, which is too many gosub calls. So to monitor it I would think I need to count the number of calls to the routine and capture that number. If the routine does not error out I'm golden, but if it does then somehow I need to put a limit on the recursions being performed using that call count. In theory then, I should be able to identify the upper recursion limit of any of my recursive sub/function routines. Correct, or would the stack have other factors which would make the call count unreliable?


@SMcNeill ... is the variable "kolor" always = 0? If not how is the value for "kolor" assigned.   I'm thinking, if the kolor value is being calculated multiple times somewhere in your code, then that multiple call is the recursive stack loading problem that I missed seeing.

 Or does the subroutine itself would fall within the scope of "Recursive" because there are multiple "if" statements? I would think multiple 'if" statements would be quite a bit less of a stack load (ie the stack doesn't have to remember as many addresses).
Title: Re: Recursion
Post by: SMcNeill on November 01, 2021, 10:13:16 am
@bplus and @luke  ... Trapping I guess would be error code 256, which is too many gosub calls. So to monitor it I would think I need to count the number of calls to the routine and capture that number. If the routine does not error out I'm golden, but if it does then somehow I need to put a limit on the recursions being performed using that call count. In theory then, I should be able to identify the upper recursion limit of any of my recursive sub/function routines. Correct, or would the stack have other factors which would make the call count unreliable?


@SMcNeill ... is the variable "kolor" always = 0? If not how is the value for "kolor" assigned.   I'm thinking, if the kolor value is being calculated multiple times somewhere in your code, then that multiple call is the recursive stack loading problem that I missed seeing.

 Or does the subroutine itself would fall within the scope of "Recursive" because there are multiple "if" statements? I would think multiple 'if" statements would be quite a bit less of a stack load (ie the stack doesn't have to remember as many addresses).

For the Fill routine, you pass it the initial color ro fill when you call the SUB and then it simply repeats the call with that same color over and over.  In the example above, the fill color is &HFFFF0000 (or _RGBA32(255, 0, 0, 255) or solid Red).   We set the fill color on line 4 with the third parameter and the routine sticks with it to completion.

I simplified the process a bit; see if it's easier to understand now.  ;D


@@@@@@
@@@@@@
@@@@@@
@@@@@@
@@@@@@
@@@@@@

Let's pretend the above is my screen and that's all black...  Now, I draw a red box on it...

RRRRRR
R@@@@R
R@@@@R
R@@@@R
R@@@@R
RRRRRR

Now, choose a point inside that Red box (let's choose point 3,3) and color it Red.

RRRRRR
R@@@@R
R@R@@R
R@@@@R
R@@@@R
RRRRRR

Now, here's where the recursion comes in:
If the points above, left, right, below that point aren't Red, repeat the process for them <-- and recursively repeat this step as many times as is needed.

FIRST RECURSION:

RRRRRR
R@R@@R
RRRR@R
R@R@@R
R@@@@R
RRRRRR

See how we filled outwards?  Repeat that fill for our next positions...

SECOND RECURSION:

RRRRRR
RRRR@R
RRRRRR
RRRR@R
R@R@@R
RRRRRR


And so on, until we're done.  ;)
Title: Re: Recursion
Post by: luke on November 01, 2021, 10:21:27 am
I don't believe stack overflow can be effectively trapped (note the GoSub system is independent). On my system I just get a segmentation fault, Windows will probably have a similarly-fatal error.

It's also important to note that the stack size can differ between computers, so what works for you might not work elsewhere.

Still, let's assume a 1MB stack (not unreasonable). There's a bit of space taken at the start for bookkeeping and runtime stuff. A subroutine call takes 8 bytes for the return address, 8 bytes for bookkeeping and then space for local variables. If we have none, that's 16 bytes per stack frame which gives up a maximum call depth of about 65 thousand (actually 2^16). I haven't tested this, and I might have missed some extra local variables that are implicitly created by QB64, but hopefully it gives you an idea of the kind of scales to expect.
Title: Re: Recursion
Post by: SMcNeill on November 01, 2021, 10:34:15 am
I don't believe stack overflow can be effectively trapped (note the GoSub system is independent). On my system I just get a segmentation fault, Windows will probably have a similarly-fatal error.

It's also important to note that the stack size can differ between computers, so what works for you might not work elsewhere.

Still, let's assume a 1MB stack (not unreasonable). There's a bit of space taken at the start for bookkeeping and runtime stuff. A subroutine call takes 8 bytes for the return address, 8 bytes for bookkeeping and then space for local variables. If we have none, that's 16 bytes per stack frame which gives up a maximum call depth of about 65 thousand (actually 2^16). I haven't tested this, and I might have missed some extra local variables that are implicitly created by QB64, but hopefully it gives you an idea of the kind of scales to expect.

Windows just crashes the program with no warning.  We don't even get a nice segment fault.

As a note, I've seen recursive paint fills die many times before.  1980x1024 = 2+ million pixels...  If you end up recursively calling to them all, you'll definitely run out of stack space.
Title: Re: Recursion
Post by: Dimster on November 01, 2021, 11:44:57 am
Thanks guys.
Title: Re: Recursion
Post by: bplus on November 01, 2021, 11:51:11 am
If you want the easiest example of recursion, then I'd like to add in a simple Fill routine for folks to study:

Code: QB64: [Select]
  1. Screen _NewImage(640, 480, 32)
  2.  
  3. Circle (240, 240), 200, &HFFFF0000
  4. Fill 240, 240, &HFFFF0000
  5.  
  6. Sub Fill (x, y, kolor As _Unsigned Long)
  7.     If Point(x, y) <> kolor Then Pset (x, y), kolor
  8.     If Point(x + 1, y) <> kolor Then Fill x + 1, y, kolor
  9.     If Point(x - 1, y) <> kolor Then Fill x - 1, y, kolor
  10.     If Point(x, y + 1) <> kolor Then Fill x, y + 1, kolor
  11.     If Point(x, y - 1) <> kolor Then Fill x, y - 1, kolor

Give it a center point.  It then paints the pixels around it, if they're not already the proper color.  Then it moves to those pixels and repeats the process.  On and on, until the region to be painted is filled in completely.

Note, this should probably have a simple IF x < 0 or y < 0 or X >= _WIDTH or y >=_HEIGHT Then EXIT SUB in it just to make certain that it doesn't try to work itself to death off-screen on us.  ;)

Weird I can't get this sucker to work?? Not so simple. I got it working for x but when I uncomment next block with y, program dies.

Code: QB64: [Select]
  1. Screen _NewImage(640, 480, 32)
  2.  
  3. Circle (240, 240), 200, &HFFFF0000
  4. Fill 240, 240, &HFFFF0000
  5.  
  6. Sub Fill (x, y, kolor As _Unsigned Long)
  7.     _Source 0
  8.     If Point(x, y) <> kolor Then
  9.         PSet (x, y), kolor
  10.         If x + 1 < _Width Then
  11.             If Point(x + 1, y) <> kolor Then Fill x + 1, y, kolor
  12.         End If
  13.  
  14.         If x - 1 >= 0 Then
  15.             If Point(x - 1, y) <> kolor Then Fill x - 1, y, kolor
  16.         End If
  17.  
  18.         ' next does not work???
  19.         'If y + 1 < _Height Then
  20.         '    If Point(x, y + 1) <> kolor Then Fill x, y + 1, kolor
  21.         'End If
  22.  
  23.         'If y - 1 >= 0 And Point(y - 1, y) <> kolor Then Fill x, y - 1, kolor
  24.     End If
  25.  

Well everything works on a way, way, way smaller radius?
Code: QB64: [Select]
  1. Screen _NewImage(640, 480, 32)
  2. _Delay .55
  3. Circle (240, 240), 70, &HFFFF0000
  4. Fill 240, 240, &HFFFF0000
  5.  
  6. Sub Fill (x, y, kolor As _Unsigned Long)
  7.     If Point(x, y) <> kolor Then
  8.         PSet (x, y), kolor
  9.         If x + 1 < _Width Then
  10.             If Point(x + 1, y) <> kolor Then Fill x + 1, y, kolor
  11.         End If
  12.  
  13.         If y + 1 < _Height Then
  14.             If Point(x, y + 1) <> kolor Then Fill x, y + 1, kolor
  15.         End If
  16.  
  17.  
  18.         If x - 1 >= 0 Then
  19.             If Point(x - 1, y) <> kolor Then Fill x - 1, y, kolor
  20.         End If
  21.  
  22.         If y - 1 >= 0 Then
  23.             If Point(x, y - 1) <> kolor Then Fill x, y - 1, kolor
  24.         End If
  25.     End If
  26.  
  27.  

I am kind of shocked. I have a Paint variation that works a heck of allot like this and I don't remember having troubles with it like I am with this. Maybe I am getting old and missing something?
Title: Re: Recursion
Post by: SMcNeill on November 01, 2021, 12:09:28 pm
@bplus that's the stack space issue at play.  Too many recursive calls with the larger radius in action.
Title: Re: Recursion
Post by: bplus on November 01, 2021, 12:14:10 pm
OK I checked back and I was not using recursion, whew!
Title: Re: Recursion
Post by: TempodiBasic on November 01, 2021, 12:48:03 pm
Recursive method is not a must, it is only a method among the others possible to use to solve a ripetitive work.

Every recursive algorythm has its equivalent as Iterative algorythm and vice versa
https://en.wikipedia.org/wiki/Recursion_(computer_science)#Recursion_versus_iteration (https://en.wikipedia.org/wiki/Recursion_(computer_science)#Recursion_versus_iteration)

so you can use recursive way if you usually do things in this way and you have a good resource machine as you can use iterative way if you usually do things in this way and you have a global view of the work to do.
I you are able to use both recursive both iterative, you have more ways to realize your task. :-)
Title: Re: Recursion
Post by: SMcNeill on November 01, 2021, 12:58:58 pm
OK I checked back and I was not using recursion, whew!

You can cut down on the recursion a ton just by logically going in a straight line each pass, rather than filling in all 4 directions a point at a time.  I just chose the most inefficient way as an easy example of the recursive method of coding.

WHILE POINT (x + 1, y) <> RED
    PSET (x + 1, y), RED
    x = x + 1
    FillVertical (x + 1, y), RED
WEND

Let's say I'm painting to the right, like above.  My WHILE loop here will fill left to right so I don't need to check those points a second time.  All I need to check is the lines leading up or down off from those points....

I'd be filling via LINE rather than via points, which would be both faster and much less recursive in the end. 
Title: Re: Recursion
Post by: TempodiBasic on November 01, 2021, 01:51:11 pm
as we can see it is  easy to go stack overflow

this a simple filling routine that uses PSET instruction used for all points of the rectangular to fill with a color
both in iterative mode both in recursive mode the module does only a PSET action at every call.

Copy, paste into our fantastic QB64 2.0.1 and press F5 to run....
Code: QB64: [Select]
  1. ' fill a square : recursive way versus iterative way
  2. ' TempoDiBasic 2021 11 01
  3.  
  4. Screen _NewImage(1200, 900, 32)
  5.  
  6. Dim Shared As Integer x1, x2, y1, y2
  7. Dim Shared As Long Backcolor, Forecolor
  8. Dim As Single t0, t1, t2
  9. Backcolor = _RGBA32(255, 127, 39, 255)
  10. Forecolor = _RGBA32(0, 125, 0, 255)
  11. x1 = 100
  12. x2 = 1100
  13. y1 = 200
  14. y2 = 700
  15.  
  16. Print "Itertive method..."
  17. t0 = Timer
  18. Cls , Backcolor
  19. fill_iterative
  20. t2 = Timer
  21. t2 = t2 - t0
  22. Print "Task iterative done in "; t2; " time"
  23.  
  24. Print "recursive method..."
  25. t0 = Timer
  26. Cls , Backcolor
  27. fill_recursive x1, y1
  28. t1 = Timer
  29. t1 = t1 - t0
  30. Print "Task recursive done in "; t1; " time"
  31. Print "resuming...."
  32. Print "Task recursive done in "; t1; " time"
  33. Print "Task iterative done in "; t2; " time"
  34.  
  35. Sub fill_recursive (x, y)
  36.     Static As Integer a, b
  37.     'Dim As Integer a, b
  38.     a = x
  39.     b = y
  40.     PSet (a, b), Forecolor
  41.     If a <= x2 Then
  42.         a = a + 1
  43.     Else
  44.         a = x1
  45.         If b <= y2 Then b = b + 1 Else Exit Sub
  46.     End If
  47.     fill_recursive a, b
  48.  
  49. Sub fill_iterative
  50.     For i = x1 To x2
  51.         For j = y1 To y2
  52.             PSet (i, j), Forecolor
  53.         Next
  54.     Next
you must have a stack overflow for recursion method also if it doesn't use other than an sum operation an IF and PSET...
the issue is not the STATIC value that you must preserve in recursion... also if you use a DIM as integer a,b you got the same stack overflow.
As Luke said the stack overflow comes out as a generic error catched by OS (Win10)
  [ This attachment cannot be displayed inline in 'Print Page' view ]  

Title: Re: Recursion
Post by: SMcNeill on November 01, 2021, 02:30:42 pm
How you folks convince Windows to toss you an error message is beyond me.  All I get is a program close when it comes to a segment fault, with no notice at all that things went wrong.  :(
Title: Re: Recursion
Post by: bplus on November 01, 2021, 02:39:24 pm
Yeah I get the run around circle and then we're back to IDE. Better than crashing though.
Title: Re: Recursion
Post by: Pete on November 02, 2021, 08:29:14 am
The QBasic keyword, FRE, was not included in QB64. FRE kept track of both memory and stack space.

FRE(-1) Reported the largest non-string array you could create.
FRE(-2) Reported unused stack space.
FRE(to the anything else) The unused free memory space for storing strings. I used to use FRE(0)

With advancements in memory and operating systems, I'd have to question if a FRE statement even could be adequately addressed in the present.

Pete
Title: Re: Recursion
Post by: Aurel on November 02, 2021, 09:17:44 am
I don't know where it is that old post in which we talk about
recursive descent parser in 3 functions ...
if this 3 functions work then RECURSION work properly.

this is the topic about recursion:
https://www.qb64.org/forum/index.php?topic=33.15
Title: Re: Recursion
Post by: Aurel on November 02, 2021, 09:36:59 am
also Steve code compile fine
but when i run it ..then crush ...
i asking why ??

Code: QB64: [Select]
  1.   Screen _NewImage(640, 480, 32)
  2.      
  3.     Circle (240, 240), 200, &HFFFF0000
  4.     Fill 240, 240, &HFFFF0000
  5.      
  6.     Sub Fill (x, y, kolor As  Long)
  7.         If Point(x, y) <> kolor Then Pset (x, y), kolor
  8.         If Point(x + 1, y) <> kolor Then Fill x + 1, y, kolor
  9.         If Point(x - 1, y) <> kolor Then Fill x - 1, y, kolor
  10.         If Point(x, y + 1) <> kolor Then Fill x, y + 1, kolor
  11.         If Point(x, y - 1) <> kolor Then Fill x, y - 1, kolor
  12.     End Sub

i changed kolor var to _offset and then program started but then crush
Title: Re: Recursion
Post by: TempodiBasic on November 02, 2021, 09:46:09 am
@SMcNeill

hi Steve
I like this "convince"
Quote
How you folks convince Windows to toss you an error message is beyond me.
I run the code posted above using QB64 x64 2.0.1, the code is already evident. The OS is Windows 8.1Pro  on HP notebook  500HDD, 4 GB RAM,  Celeron 2.00Ghz CPU N2810.
(here compiling is a bit slower than other my notebooks)
The application windows disappears and Windows tosses that generic error message. And I presume that it is related to a stack overflow.
But as you said, Windows shows a generic error message and it is impossible to have back an error code.

@ very interesting FRE of QB45. I think that it is hard to have a similar function in Windows.
Title: Re: Recursion
Post by: TempodiBasic on November 02, 2021, 10:07:07 am
@Aurel
hi dear
the QB64 2.0 was born because developers have solved the recursive bug
see here how the old broken code works good now
  [ This attachment cannot be displayed inline in 'Print Page' view ]  

https://www.qb64.org/portal/changelog-for-v2-0/ (https://www.qb64.org/portal/changelog-for-v2-0/)
https://www.qb64.org/forum/index.php?topic=4209.0 (https://www.qb64.org/forum/index.php?topic=4209.0)
Title: Re: Recursion
Post by: SMcNeill on November 02, 2021, 10:19:14 am
also Steve code compile fine
but when i run it ..then crush ...
i asking why ??

Code: QB64: [Select]
  1.   Screen _NewImage(640, 480, 32)
  2.      
  3.     Circle (240, 240), 200, &HFFFF0000
  4.     Fill 240, 240, &HFFFF0000
  5.      
  6.     Sub Fill (x, y, kolor As  Long)
  7.         If Point(x, y) <> kolor Then Pset (x, y), kolor
  8.         If Point(x + 1, y) <> kolor Then Fill x + 1, y, kolor
  9.         If Point(x - 1, y) <> kolor Then Fill x - 1, y, kolor
  10.         If Point(x, y + 1) <> kolor Then Fill x, y + 1, kolor
  11.         If Point(x, y - 1) <> kolor Then Fill x, y - 1, kolor
  12.     End Sub

i changed kolor var to _offset and then program started but then crush

The crash you're seeing is what happens if you try to do too much recursion -- you run out of stack space and crash.  Change the CIRCLE radius from 200 to something like 50 and then try it and the code will run fine. 

Different computers and different OSes have different amounts of stack space associated with them.  MODERN Windows defaults to only 1MB, while Linux is 8MB, if my memory is right, but older versions had lower limits and those values can be altered in different manners on different OSes and compilers... 

Which leads to a serious limit and issue with recursion -- how much is too much?  How many times can a routine call itself before crashing? 

That's hard to say.  Just because it works on YOUR system, it doesn't mean it will on somebody else's.  Limit recursive functions to small calls and be careful not to run out of the limited stack space you have.  As you're experiencing yourself, the crashes in Windows come with no warning and no diagnostic message.  Just POOF!!  Program closed!!

Everyone who codes always seems obsessed with the idea of recursive code, because of its simplicity (doesn't get much simpler than the fill routine above) -- but they also need to be aware of the serious limits which comes with such style coding. 

Honestly, in most cases, I'd personally try NOT to use recursion in my code, if I can avoid it.  However, there's times where its ease of implementation and limited use capacity aren't a big deal, and I'll toss it in something just to get it done and over with quickly.  In my mind, recursive routines go into the same category as BubbleSorts and GOTO statements -- generally best to avoid like the plague, but secretly used anyway in private when being lazy.

Just be aware of the limits when writing any type os sub/function/gosub that might end up calling itself repeatedly.
Title: Re: Recursion
Post by: Dimster on November 02, 2021, 10:53:25 am
In terms of the load on the stack: Lets say you have a program of a couple of thousand lines and in the main section there are multiple opening and closing files capturing tons of data and placing that in arrays. Then there are many many Do Loops with multiple If statements and Select Cases. Your program runs with no stack issues. Now suppose you can see where the lines of code can be reduced by eliminating the Do Loops with Recursive Subs or Functions. Would it be safe to conclude the effect of stack load is the same (ie there should be no issue)?
Title: Re: Recursion
Post by: SMcNeill on November 02, 2021, 11:11:28 am
In terms of the load on the stack: Lets say you have a program of a couple of thousand lines and in the main section there are multiple opening and closing files capturing tons of data and placing that in arrays. Then there are many many Do Loops with multiple If statements and Select Cases. Your program runs with no stack issues. Now suppose you can see where the lines of code can be reduced by eliminating the Do Loops with Recursive Subs or Functions. Would it be safe to conclude the effect of stack load is the same (ie there should be no issue)?

Not at all.  Your program is limited by OS memory.  On 32-bit systems that's around 1.5GB  of memory after system reserved stuff, and on 64-bit systems it's limited by your machine's physical memory.  Stack space is limited to ~ 1MB on Windows and ~ 8 MB on Linux.

That's an incredible reduction in total amount of memory available.  Recursion must always be used in decisive moderation.
Title: Re: Recursion
Post by: Dimster on November 02, 2021, 02:50:59 pm
Would CLS clear the stack? If not, is there another command or routine which clears the stack. I would think the completion of the recursive sub or function clears the stack upon return to the main program for a fresh set of addresses but is that the only way the stack resets?
Title: Re: Recursion
Post by: Aurel on November 02, 2021, 03:32:25 pm
heh yes
with 20 work!
even i am not sure that i understand it exactly how is connected with
circle radius( stupid me ..he he )..
Title: Re: Recursion
Post by: TempodiBasic on November 02, 2021, 07:19:54 pm
@SMcNeill
I get crushing and this graphic output
with this code (note that radius is 30 and not 200)
Quote
Code: QB64: [Select]
  Screen _NewImage(640, 480, 32)
     
    Circle (240, 240), 30, &HFFFF0000
    Fill 240, 240, &HFFFF0000
     
    Sub Fill (x, y, kolor As  Long)
        If Point(x, y) <> kolor Then Pset (x, y), kolor
        If Point(x + 1, y) <> kolor Then Fill x + 1, y, kolor
        If Point(x - 1, y) <> kolor Then Fill x - 1, y, kolor
        If Point(x, y + 1) <> kolor Then Fill x, y + 1, kolor
        If Point(x, y - 1) <> kolor Then Fill x, y - 1, kolor
    End Sub

  [ This attachment cannot be displayed inline in 'Print Page' view ]  

I need to put
Code: QB64: [Select]
after SCREEN to be able to see window of program.

@Aurel
do you like to joke, surely you know that circle area = radius * radius * 3,14...........
have a fun with ricursion coding.
Title: Re: Recursion
Post by: bplus on November 02, 2021, 07:38:36 pm
Would CLS clear the stack? If not, is there another command or routine which clears the stack. I would think the completion of the recursive sub or function clears the stack upon return to the main program for a fresh set of addresses but is that the only way the stack resets?

No CLS would not clear stack. Yes, just exiting the routine would clear the stack. I even tried that before we reached the 10,8xx number with that notorious Fill routine, almost worked. I managed to hack another cycle of Fill, as mentioned in the other thread.
Title: Re: Recursion
Post by: Dimster on November 03, 2021, 02:26:02 pm
So we need a recursion call of the recursive sub/function .... I'm smiling, clearly this spells disaster. But, if you do not have a termination case for the recursion, and have it housed within a Do Loop which makes control calls to the recursive sub/function, then would the stack be cleared?

For Example
a=1000
b=a
c=a
Do
    Do
         Do
         CALL Recursive Sub/Function
          Loop Until a
    CALL Recursive Sub/Function
    Loop Until b
CALL Recursive Sub/Function
Loop Until c

As long as a,b or c have clear clean stacks at the beginning of their loops, then perhaps this method could handle larger numbers of address calls. Kinda combersome.

Title: Re: Recursion
Post by: bplus on November 03, 2021, 05:45:50 pm
@Dimster

It just so happens I was thinking about this last night. A recursive routine can act as a loop.

Here is a quick Demo:
Code: QB64: [Select]
  1. StartIndex = 2: stepperIndex = 2: DoneIndex = 20
  2. For index = StartIndex To DoneIndex Step stepperIndex
  3.     Print index ^ 2;
  4. index = StartIndex
  5. PrintSquare index, stepperIndex, DoneIndex
  6.  
  7. ' can be done recursively
  8. Sub PrintSquare (index, stepper, indexDone)
  9.     If index > indexDone Then Exit Sub
  10.     Print index ^ 2;
  11.     _Delay .1
  12.     index = index + stepper
  13.     PrintSquare index, stepper, indexDone
  14.  
  15.  

Actually I had a little trouble with trying it this way:
Code: QB64: [Select]
  1. StartIndex = 2: stepperIndex = 2: DoneIndex = 20
  2. For index = StartIndex To DoneIndex Step stepperIndex
  3.     Print index ^ 2;
  4. index = StartIndex
  5. PrintSquare index, stepperIndex, DoneIndex
  6.  
  7. ' can be done recursively
  8. Sub PrintSquare (index, stepper, indexDone)
  9.     If index > indexDone Then Exit Sub
  10.     Print index ^ 2;
  11.     PrintSquare index + stepper, stepper, indexDone
  12.  

Oh heck it works that way too. OK so it was before supper blood sugar low = good chance I spelled stepper wrong!

Title: Re: Recursion
Post by: Dimster on November 04, 2021, 09:34:30 am
Wow @bplus. I ran both versions with the termination case at 2,000,000 and they ran quickly with no hint of a stack over load issue. Thanks for this, I think these will be my templates. Very similar structure to Quick Sort.

The recursive sub I was working with had an input statement to grab a data item from a file and then I had a naked recursive call (ie no parameters). I had the termination case just before the naked call using the end of file. Looked to me that that should have worked but never could get it to run to the end of the file. Going to revamp using your routines as my template. Thanks again.
Title: Re: Recursion
Post by: bplus on November 04, 2021, 01:53:50 pm
@Dimster

I hope you practice a little with recursive routines before attempting something serious ie with filed data.
They are simple, elegant usually, but they can be tricky to get working.

If you can replace a Recursive subroutine with a simple loop that would very likely be a better way to go.
Title: Re: Recursion
Post by: TempodiBasic on November 04, 2021, 08:07:28 pm
Quote
A recursive routine can act as a loop
Yes a loop in which you take track of previouse value and actions done in the recursive block!
It becames too mad when you call the sub at the middle of itself, leaving undone the second part of the SUB/FUNCTION

see here what difference with the little addition of a line of code to the 2nd example of Bplus

Code: QB64: [Select]
  1. StartIndex = 2: stepperIndex = 2: DoneIndex = 20
  2. For index = StartIndex To DoneIndex Step stepperIndex
  3.     Print index ^ 2;
  4.     _Delay .1
  5.     If index = 18 Then stepperIndex = -2  '<------------------------newline
  6. Print "--------------------------------------"
  7. index = StartIndex
  8. PrintSquare index, stepperIndex, DoneIndex
  9.  
  10. ' can be done recursively with much caution to control the effect
  11. Sub PrintSquare (index, stepper, indexDone)
  12.     If index > indexDone Then Exit Sub
  13.     Print index ^ 2;
  14.     _Delay .1
  15.     If index = 18 Then stepper = -2  '<-----------------------newline
  16.     PrintSquare index + stepper, stepper, indexDone
  17.  
  18.  
Title: Re: Recursion
Post by: bplus on November 04, 2021, 08:43:28 pm
see here what difference with the little subtraction in a line of code to the 1st example of Bplus mod by TempodiBasic
Code: QB64: [Select]
  1. StartIndex = 2: stepperIndex = 2: DoneIndex = 20
  2. For index = StartIndex To DoneIndex Step stepperIndex
  3.     Print index ^ 2;
  4.     _Delay .1
  5.     If index = 18 Then index = -2 '<------------------------newline
  6.  
Title: Re: Recursion
Post by: Dimster on November 05, 2021, 09:55:17 am
Good advice @bplus . I am interested in improving the speed of some of my looping routines in the main program code. I'm hoping to see a difference in speed from the iteration of a loop in the main program to recursion in a sub/function. Iteration and recursion seem to mean the same thing to me but as I'm looking at my code the iteration of looping seems more cumbersome (less elegant to use your observation) than recursive coding. But as  you say, I'll "practice a little" with it and see if I can achieve either better accuracy or improved speed with recursion. Truth be told, can't hurt to learn recursion.

@TempodiBasic ... I gather the line you added has the effect of not only creating an infinite loop but also a demonstration that the loop will never overload the stack. But is the bigger point you are making that once recursion is used, you can stop it's run and restart from where it left off whereas Looping, once stopped, needs to restart from the beginning?? Sorry if I misunderstood. 
Title: Re: Recursion
Post by: bplus on November 05, 2021, 06:24:17 pm
@Dimster

Here's another nice one for your collection:
Code: QB64: [Select]
  1. _Title "GCD by Recursion" 'b+ 2021-11-05
  2. ' In mathematics GCD or Greatest Common Divisor of two or more integers
  3. ' is the largest positive integer that divides both the number
  4. ' without leaving any remainder.
  5.  
  6.     Input "enter a, b or 0 to quit "; a, b
  7.     If a <> 0 And b <> 0 Then Print GCD(a, b) Else End
  8.  
  9.     If b = 0 Then GCD = a Else GCD = GCD(b, a Mod b)
  10.  
Title: Re: Recursion
Post by: TempodiBasic on November 05, 2021, 10:59:29 pm
@bplus
fine as you have make a pendulum (infinite palindrome generator) of digit sequence!
It is very interesting and fine! The power of 2 is very strong!

--------
@bplus @Dimster
what do you think about the FOR STEP NEXT glitch pointed out by my example?
Code: QB64: [Select]
  1. StartIndex = 2: stepperIndex = 2: DoneIndex = 20
  2. Print "StepperIndex is 2 but when index will be 18 it becomes -2"
  3. Print " so we expect to see index to decrement after 18 towards negative digits"
  4. For index = StartIndex To DoneIndex Step stepperIndex
  5.     Print index; "->"; index ^ 2;
  6.     _Delay .1
  7.     If index = 18 Then stepperIndex = -2 '<------------------------newline
  8.  
  9. Print "another try: the variable a grows up to 100 step 10 until"
  10. Print " it becomes 50, after this variable a should decrease step -10"
  11. a = 1: b = 100: c = 10
  12. For a = 1 To b Step c
  13.     Print a; " ";
  14.     If a = 50 Then c = -c
  15.  


@Dimster
yes you are very near to original sight of my communication
Loop and recursion can be very similar but often a change made into a loop make a very different effect if it is made in a recursion loop.
This is more true when the recursive loop call itself in the middle of sub/function recursive because the rest part of recursive has not been already executed. So you cannot have a more specific control on this part of recursion in the meaning that you cannot know what recursion bring back as true as you cannot know how times the recursion will be called on fly.
From this comes out that you say in your affimation: if you stop a recursion the data in all the sub called are still available, while in the loop you can access only to the data of the current cycle of loop.
But yes recursion is fine, beautyful, elegant and more exciting versus loop when it is possible, you must only pay more attention for details.
see this financial example
Code: QB64: [Select]
  1. $Debug
  2. 'recursion versus loop
  3. ' calcularion of compound interest of a capital in years of investment
  4. ' here a definition: https://en.wikipedia.org/wiki/Future_value#Compound_interest
  5.     choice$ = ""
  6.     Print " calculation of compound interest of a capital"
  7.     Print " please type amount of money, annual rate (rate for year), years of investment>"
  8.     Input "", money!, AnnRate!, Years%
  9.     Print CompoundInterest_loop
  10.     Print compoundInterest_Recursive(money!, AnnRate!, Years%)
  11.     Print "    press Spacebar to end, another key to replay"
  12.     Do While choice$ = ""
  13.         choice$ = InKey$
  14.     Loop
  15. Loop Until choice$ = " "
  16.  
  17. Function CompoundInterest_loop
  18.     Shared money!, AnnRate!, Years%
  19.     newmoney! = money! ' we use a local variable to not modify main variable MONEY!
  20.     For year = 1 To Years% Step 1
  21.         newmoney! = newmoney! + (newmoney! * AnnRate! / 100)
  22.     Next
  23.     CompoundInterest_loop = newmoney!
  24.  
  25. Function compoundInterest_Recursive (newmoney!, Rate!, ActYears%)
  26.     compoundInterest_Recursive = newmoney! ' <------------- this save last result
  27.     If ActYears% = 0 Then Exit Function
  28.     Themoney! = newmoney! + ((newmoney! * Rate!) / 100)
  29.     ActYears% = ActYears% - 1
  30.     money! = compoundInterest_Recursive(Themoney!, Rate!, ActYears%)
  31.     compoundInterest_Recursive = money!

-----------------------------------------------

@FellippeHeitor  and @QB64 Developers Team

nice to do some sessions of debug.
I find this new feature very powerful.
Waiting when I can observe array variable with variable index.
Thanks for your efforts.



Title: Re: Recursion
Post by: bplus on November 05, 2021, 11:51:02 pm
Quote
@bplus @Dimster
what do you think about the FOR STEP NEXT glitch pointed out by my example?

Code: QB64: [Select]
  1. Print "another try: the variable a grows up to 100 step 10 until"
  2. Print " it becomes 50, after this variable a should decrease step -10"
  3. a = 1: b = 100: c = 10
  4. For a = 1 To b Step c
  5.     Print a; " ";
  6.     If a = 50 Then c = -c ' <<<<<<<<<<<<< a never = 50 so c never changed!
  7. Print: Print "c ="; c
  8.  
  9. Print "another try: the variable a grows up to 100 step 10 until"
  10. Print " it becomes 50, after this variable a should decrease step -10"
  11. a = 1: b = 100: c = 10
  12. For a = 1 To b Step c
  13.     Print a; " ";
  14.     If a = 51 Then c = -c 'fixed but in For loop b, c changes aren't acknowledged until after loop
  15. Print: Print "c ="; c
  16.  
  17. Print "another try: the variable a grows up to 10 step 1 until"
  18. Print " it becomes 5, after this variable a should decrease step -1"
  19. a = 1: b = 10: c = 1
  20. For a = 1 To b Step c
  21.     c = c + 1 ' change c into counter
  22.     Print a; " ";
  23.     If a = 5 Then b = 9 ' !!!! For loop b, c changes aren't acknowledged until after loop
  24.     If b = 9 Then a = a - 2
  25.     If a = 0 Then b = 0 ' So we exploit that and make "palindromic pendulum"
  26.     _Delay .5
  27. Print: Print "c ="; c
  28.  
  29.  
Title: Re: Recursion
Post by: Dimster on November 06, 2021, 10:32:00 am
@TempodiBasic - thanks for that Compound Interest routine. It's a handy little item especially for a person like myself who follows the stock market.

@bplus - GCD has potential application in some of the data crunching I'm working with, thanks for same. The question I have on that routine is , How would it meet the definition of Recursion as we have been discussing here? I understand Recursion is a sub/function which calls itself but there does not appear to be a specific call to itself in the GCD code.

On the matter of the negative stepping. Originally it struck me that the iterations could never reach the DoneIndex @ 20, making it an endless loop. I totally missed the bigger picture of the mis-stepping when the negative stepping kicked in.  Some very good lessons here.

One other question @bplus - I notice you initialized the variables before each looping. Is there an advantage to this or just because there were multiple values in "a" prior to the looping? Especially when variable "a" which gets its values in the For statement so the prior values should be overwritten??
Title: Re: Recursion
Post by: bplus on November 06, 2021, 11:00:09 am
Quote
@bplus - GCD has potential application in some of the data crunching I'm working with, thanks for same. The question I have on that routine is , How would it meet the definition of Recursion as we have been discussing here? I understand Recursion is a sub/function which calls itself but there does not appear to be a specific call to itself in the GCD code.

but GCD is calling itself, look again:
Code: QB64: [Select]
  1.     If b = 0 Then GCD = a Else GCD = GCD(b, a Mod b)
  2.  
  3. '' Else GCD = GCD(b, a Mod b)" that's a call to itself.
  4.  

Quote
One other question @bplus - I notice you initialized the variables before each looping. Is there an advantage to this or just because there were multiple values in "a" prior to the looping? Especially when variable "a" which gets its values in the For statement so the prior values should be overwritten??

Oh the a = 1, that's just lazy copy/paste, you are right a = 1 was waste of space because a is reset in For loop.
Title: Re: Recursion
Post by: bplus on November 06, 2021, 11:44:29 am
@TempodiBasic

+1 agreed that is a handy calculation.

I have made a couple of mods to maybe clear up wording and show elegance of recursive function:
Code: QB64: [Select]
  1. ' bplus mod of TempodiBasic 2021-11-06
  2. 'recursion versus loop
  3. ' calcularion of compound interest of a capital in years of investment
  4. ' here a definition: https://en.wikipedia.org/wiki/Future_value#Compound_interest
  5.     choice$ = ""
  6.     ' bplus changed wording of next 3 lines
  7.     Print " Calculation of Compound (Yearly) Interest of a Principle Deposit."
  8.     Print " Please type: (don't forget commas between numbers)"
  9.     Print " Principle amount of money, Annual Rate Percent, Years of investment"
  10.     Input "", money!, AnnRate!, Years%
  11.     Print CompoundInterest_loop
  12.     Print compoundInterest_Recursive(money!, AnnRate!, Years%)
  13.     Print "    press Spacebar to end, another key to replay"
  14.     Do While choice$ = ""
  15.         choice$ = InKey$
  16.     Loop
  17. Loop Until choice$ = " "
  18.  
  19. Function CompoundInterest_loop
  20.     Shared money!, AnnRate!, Years% ' bplus comments nice! unfamiliar with this use of Shared
  21.     newmoney! = money! ' we use a local variable to not modify main variable MONEY!
  22.     For year = 1 To Years% Step 1
  23.         newmoney! = newmoney! + (newmoney! * AnnRate! / 100)
  24.     Next
  25.     CompoundInterest_loop = newmoney!
  26.  
  27. 'Function compoundInterest_Recursive (newmoney!, Rate!, ActYears%)
  28. '    compoundInterest_Recursive = newmoney! ' <------------- this save last result  ??? do we need this ???
  29. '    If ActYears% = 0 Then Exit Function
  30. '    Themoney! = newmoney! + ((newmoney! * Rate!) / 100)
  31. '    ActYears% = ActYears% - 1
  32. '    money! = compoundInterest_Recursive(Themoney!, Rate!, ActYears%) ' ??? do we need this ???
  33. '    compoundInterest_Recursive = money!
  34. 'End Function
  35.  
  36. ' bplus tries some line removals
  37. Function compoundInterest_Recursive (newmoney!, Rate!, ActYears%)
  38.     If ActYears% = 0 Then compoundInterest_Recursive = newmoney!: Exit Function
  39.     compoundInterest_Recursive = compoundInterest_Recursive(newmoney! + ((newmoney! * Rate!) / 100), Rate!, ActYears% - 1)
  40.  
  41.  
  42.  
Title: Re: Recursion
Post by: Dimster on November 06, 2021, 01:43:38 pm
I see it now @bplus - said the blind man as he picked up his hammer and saw.
Title: Re: Recursion
Post by: FellippeHeitor on November 06, 2021, 04:23:18 pm
@TempodiBasic hi Tempo. Regarding:

Quote
Waiting when I can observe array variable with variable index

That's already a feature. When you double click an array in the Variable List, you will be asked which indexes you want to watch.
Title: Re: Recursion
Post by: TempodiBasic on November 06, 2021, 10:53:34 pm
@bplus
Hi, I like your beautify of recursive function. Very good!

@Dimster
I'm happy that I can give a positive spin to your curiosity and interest about recursion method of programming

@FellippeHeitor
Hi,
maybe that I have misunderstood how to watch into arrays. I'll watch again your video on youtube.
For now I have understood (wrongly) that I can observe a(1) or a(19) of A(1 to 20) but not a(b%) also in the specific
row of code in which b% has its clear value.
Thanks
Title: Re: Recursion
Post by: FellippeHeitor on November 07, 2021, 12:17:41 am
@TempodiBasic yeah, you can't do that. There are no plans for now to implement that either.