Recursion: Avoiding Stack Overflow Errors

AI Thread Summary
The discussion centers on the limitations of recursion in algorithms, particularly regarding stack overflow errors. The maximum levels of recursion depend on the available stack space and the space each call consumes, which varies by programming language, environment, and operating system. While some advocate for avoiding recursion in favor of iterative solutions, others highlight the potential of tail recursion, which can be optimized by certain compilers to function similarly to loops, thus conserving stack space. However, not all languages support tail call optimization, leading to differing opinions on its effectiveness. Examples like factorial and GCD illustrate the nuances of recursion, with factorial being non-tail recursive unless modified. The conversation emphasizes the importance of choosing the right algorithm to prevent stack overflow and the need for code to function correctly during debugging, regardless of compiler optimizations.
petrushkagoogol
Messages
28
Reaction score
4
Upto how many levels of recursion can be used in an algorithm to avoid stack overflow error ?
 
Technology news on Phys.org
There is no guarantees as it depends upon:
1. How much stack space is there to begin with?
2. How much space does each call take?

While you can measure these for a particular use case, in general you can't rely on this.

For any of the products I work on, I try to avoid using recursion and use an iterative algorithm. When I do use recursion, it is either for an internal tool that customers won't have to use or has some means of guaranteeing it won't require a deep stack.
 
  • Like
Likes Fooality and Pepper Mint
petrushkagoogol said:
Upto how many levels of recursion can be used in an algorithm to avoid stack overflow error ?

This question could not be more abstract. What kind of recursion, which programming environment / language and last but not least which operating system?
 
It depends.
For example, tail recursion is as good as any iteration.
 
Irene Kaminkowa said:
It depends.
For example, tail recursion is as good as any iteration.
I don't think so. In recursion of any kind, the return address and any parameters have to be pushed onto the stack, something that is not done when a loop executes. Eventually the program will run out of available stack storage.
 
  • Like
Likes QuantumQuest
It depends.
A compiler (interpreter) may treat tail-recursive calls as jumps not function calls. Then, the space is constant.
 
Irene Kaminkowa said:
It depends.
A compiler (interpreter) may treat tail-recursive calls as jumps not function calls. Then, the space is constant.
If the recursive function takes parameters, I don't see how a compiler could rewrite a function call as a jump. The compiler would still have to push the parameters onto the stack.
 
  • Like
Likes QuantumQuest
Irene Kaminkowa said:
It depends.
A compiler (interpreter) may treat tail-recursive calls as jumps not function calls. Then, the space is constant.
Exactly. Many compilers are very good at recognizing tail recursion. It's called "Tail call optimization". Some interpreters are also good at recognizing tail recursion (e.g., many dialects of Lisp), many others are not (python (Guido hates TCO), perl, ruby, ...).
 
Mark44 said:
If the recursive function takes parameters, I don't see how a compiler could rewrite a function call as a jump. The compiler would still have to push the parameters onto the stack.
Tail recursion is a very special kind of recursion. The following C function to compute n! is recursive but it is not tail recursive:
C:
unsigned factorial (unsigned n) {
    if (n <= 1) {
        return 1;
    } else {
        return n * factorial(n-1);
    }
It appears to be tail recursive because the last function that is called is the function itself. However, the last operation is to multiply the value returned value by that call with n. One needs to write a helper function to make this tail recursive in C:
C:
unsigned factorial_helper (unsigned n, unsigned prod) {
    if (n <= 1) {
        return prod;
    } else {
        return factorial_helper(n-1, n*prod);
    }
}

unsigned factorial (unsigned n) {
    return factorial_helper(n, 1);
}

Functional languages invite one to write tail recursive functions in lieu of loops. Non-functional languages such as C (and everything derived from C) invite one to write loops in lieu of tail recursive functions.
 
  • Like
Likes FactChecker, Pepper Mint and Irene Kaminkowa
  • #10
Mark44 said:
The compiler would still have to push the parameters onto the stack
Consider GCD - it can be implemented as tail recursion.
Factorial - cannot.

There is an article "Tail call" in Wiki on this subject.
 
  • #11
Irene Kaminkowa said:
Consider GCD - it can be implemented as tail recursion.
Factorial - cannot.
I just showed one for factorial. It can be implemented as tail recursion.

Ackermann's function cannot.
 
  • #12
Good point. My mindset is from learning C 30+ years ago, before there were such sophisticated optimization schemes.
 
  • #13
Perhaps, there is a toggle for each recursion.
Mathematicians have not proved the opposite so far.
So, there is hope. And a scope for work.
 
  • #14
Usually if you even come close to a stack overflow, you are using the wrong algorithm. What kind of algorithm are you doing?

Also, why is everyone talking about compiler optimization? The code has to work while he's debugging it too, without the compiler optimizations.
 
  • Like
Likes petrushkagoogol

Similar threads

Back
Top