What
is Recursion?
In
order to say exactly what recursion is, we first have to answer "What is
recursion?" Basically, a function is said to be recursive if it calls
itself. Below is pseudocode for a recursive function that prints the phrase
"Hello World" a total of count times:
function
HelloWorld(count)
{
if(count<1)return
print("Hello World!")
HelloWorld(count - 1)
}
It might not be
immediately clear what we're doing here - so let's follow through what happens
if we call our function with count set to 10. Since count is
not less than 1, we do nothing on the first line. On the next, we print
"Hello World!" once. At this point we need to print our phrase 9 more
times. Since we now have a HelloWorld function that can do just that, we simply
call HelloWorld (this time with count set to 9) to print the
remaining copies. That copy of HelloWorld will print the phrase once, and then
call another copy of HelloWorld to print the remaining 8. This will continue
until finally we call HelloWorld with count set to zero. HelloWorld(0)
does nothing; it just returns. Once HelloWorld(0) has finished, HelloWorld(1)
is done too, and it returns. This continues all the way back to our original
call of HelloWorld(10), which finishes executing having printed out a total of
10 "Hello World!"s.
Why use Recursion?
The problem we
illustrated above is simple, and the solution we wrote works, but we probably
would have been better off just using a loop instead of bothering with
recursion. Where recursion tends to shine is in situations where the problem is
a little more complex. Recursion can be applied to pretty much any problem, but
there are certain scenarios for which you'll find it's particularly helpful. In
the remainder of this article we'll discuss a few of these scenarios and, along
the way, we'll discuss a few more core ideas to keep in mind when using
recursion.
Example: Think about computing n! recursively. I still don't know what and how my function will do this. And I also don't know, like what could be 5! or 7!... But nothing to worry about, I know that 0! = 1! = 1. And I also know that, n! = n(n-1)!. So I will think like that, "I will get n! from a function F, if some one gives me (n-1)!, then I'll multiply it with n and produce results". And, thus, F is the function for computing n!, so why don't I use again to get (n-1)! ? And when F tries to find out what could be the value of (n-1)!, it also stumbles at the same point, it wonders what would be the value of (n-2)!... then we tell him to use F again to get this... and this thing keeps going as long as we don't know what F actually should return for any k. In case of k = 1, as we know now F can return 1 for k = 1, and it don't need to call another F to solve anything first...
Example: Think about computing n! recursively. I still don't know what and how my function will do this. And I also don't know, like what could be 5! or 7!... But nothing to worry about, I know that 0! = 1! = 1. And I also know that, n! = n(n-1)!. So I will think like that, "I will get n! from a function F, if some one gives me (n-1)!, then I'll multiply it with n and produce results". And, thus, F is the function for computing n!, so why don't I use again to get (n-1)! ? And when F tries to find out what could be the value of (n-1)!, it also stumbles at the same point, it wonders what would be the value of (n-2)!... then we tell him to use F again to get this... and this thing keeps going as long as we don't know what F actually should return for any k. In case of k = 1, as we know now F can return 1 for k = 1, and it don't need to call another F to solve anything first...
nt factorial( int n) { |
| // I know this, so I don't want my function to go any further... |
| if (n==0) return 1; |
| // don't bother what to do, just reuse the function... |
| else return n*factorial(n-1); |
| } |
#2: What ever you can do with loops, can be done with recursions as well. A simple technique for converting recursions and loops is shown below:
Forward:
for loop:
1 | for ( int i = 0; i < n; i++) { |
2 | // do whatever needed |
3 | } |
Equivalent recursion:
1 | void FOR( int i, int n) { |
2 | if (i==n) return ; // terminates |
3 | // do whatever needed |
4 | FOR(i+1, n); // go to next step |
5 | } |
Backward:
for loop:
1 | for ( int i = n-1; i >= 0; i -= 1) { |
2 | // do whatever needed |
3 | } |
Equivalent recursion:
Well, you may wonder how this is backward loop? But just think of its execution cycle, just after entering the function, it is calling itself again incrementing the value of i, and the execution routine that you have written under the function call, was paused there. From the new function it enters, it works the same way, call itself again before executing anything...Thus when you have reached the limiting condition, or the base condition, then the function stops recursion and starts returning, and the whole process can be shown as below...let n=5, and we want to print 5 4 3 2 1...code for this might be:
1 | void function( int i, int n) { |
2 | if (i<=n) { |
3 | function(i+1, n); |
4 | printf ( "%d " , i); |
5 | } |
6 | } |
Explanation might look like this:
01|call function1 with i=1
02| call function2 with i=2
03| call function3 with i=3
04| call function4 with i=4
05| call function5 with i=5
06| call function6 with i=6
07| i breaks condition, no more calls
08| return to function5
09| print 5
10| return to function4
11| print 4
12| return to function3
13| print 3
14| return to function2
15| print 2
16| return to function1
17| print 1
18|return to main, done!
Left side number shows the execution steps, so from the above program, we get, 5 4 3 2 1. So indeed it ran on reverse direction...
#3: Use the advantage of call stack. When you call functions recursively, it stays in the memory as the following picture demonstrates.
int f(int n) { if(n==0) return 1; return n*f(n-1); }
You know about stack, in a stack, you cannot remove any item unless it is the topmost. So you can consider the calling of a recursive function as a stack, where, you can't remove the memory used by f(n=3) before removing f(n=2) and so so... So, you can easily see that, the functions will hold all their variables and values until it is released. This actually serves the purpose of using an array.
No comments:
Post a Comment