Recursion solves problems where the solution depends on solutions to smaller instances of the same problem (see divide and conquer) by using functions that call themselves from within their own code.
The power of recursion evidently lies in the possibility of defining an infinite set of objects by a finite statement. In the same manner, an infinite number of computations can be described by a finite recursive program, even if this program contains no explicit repetitions.
When you write a recursive function, you have to tell it when to stop recursing. Every recursive function has two parts: the base case and the recursive case so that it doesn't recurse forever.
// Countdown example
function countdown(n) {
console.log(n);
if (n < 1) {
return; // base case
} else {
countdown(n-1); // recursive case
}
}
- Head (or top) recursion: the recursive call is the first thing to happen in the function. The base case is evaluated last.
- Tail (or bottom) recursion: the recursive call is the last thing to happen in the function. The base case is evaluated first.
- Body recursion: the recursive call is in the middle of the function. The base case is evaluated first.
Usually involves a stack.
// Factorial example
function factorial(n) {
if (n === 0) {
return 1;
} else {
return n * factorial(n-1);
}
}
Every recursive program can be written using iterative methods. Recursion is used when it makes the solution clearer.
Loops may chieve a performance gain for your program. Recursion may achieve a performance gain for your programmer. Choose which is more important in your situation. - Leigh Caldwell.
Iterative approaches will often use a stack or queue to store values while they are waiting to be processed. Recursion does this implicitly, because the function call stack is used.
Say you have a box containing more boxes, with more boxes inside those boxes. There's a unique box that contains a diamond. How do you find the diamond?
- Make a pile of boxes to look through.
- Grab a box and look through it.
- If you find a box, add it to the pile to look through later.
- If you find a key, you're done.
- Repeat.
function lookForKey(box) {
const pile = [box];
while (pile.length > 0) {
const box = pile.shift();
for (const item of box) {
if (item === 'key') {
console.log('found the key!');
return;
} else {
pile.push(item);
}
}
}
}
- Look through the box.
- If you find a box, go to step 1.
- If you find a key, you're done.
function lookForKey(box) {
for (const item of box) {
if (item === 'key') {
console.log('found the key!');
return;
} else {
lookForKey(item);
}
}
}