The Fibonacci sequence is a sequence: 0, 1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89, 144…… The sequence starts with the third term, and each term is equal to the sum of the first two terms.
recursive
const fib1 = function(n) { if (n > 1) return fib1(n - 1) + fib1(n - 2); return n; }; fib1(1000); // my MacPro doesn't workCopy the code
ES6 recursive
const fib1 = n => (n > 1 ? fib1(n - 1) + fib1(n - 2) : n); fib1(1000); // Same as aboveCopy the code
Why can’t my computer come up with the results?
I changed n to 10, and I printed the number of recursive calls to fib1, which is 177;
I changed n to 20, and I printed out the number of recursive calls to fib1, which is 21,891;
So if n is equal to 1,000, how many times do I have to call it? Let me know in the comments section.
So when I was printing, I saw that there were a lot of repeated calls, like fib of one, which might be called multiple times, so can we cache the value of fib of one?
If we’ve already evaluated fib of one, we’re going to cache it in a cache variable, and then the next time we call it, we’re just going to pull it out of the cache! So let’s get started!
Optimal recursive
const memozi = (_fib) => { const cache = {}; return (n) => { if (cache[n] === undefined) { cache[n] = _fib(n); } return cache[n]; }; }; const fib2 = memozi(n => n > 1 ? fib2(n - 1) + fib2(n - 2) : n); fib2(1000); / / 4.346655768693743 e+208Copy the code
I tried to optimize the recursive call using caching, and the result is 4.346655768693743e+208
N = 10, 11 calls
N = 20, 21 calls
So if n is equal to 1,000, it must have been called 1001 times.
Then, after our optimization, the result of n = 1000 can be quickly calculated, and the calculation time is about 0.4ms.
The optimized recursion, it’s already pretty fast, so is there any other way to do it faster?
The for loop
const fib3 = (n) => { let n1 = 0; let n2 = 1; let c = n1; for (let i = 1; i <= n; i++) { c = n1 + n2; n2 = n1; n1 = c; } return c; } fib3(1000); / / 4.346655768693743 e+208Copy the code
This code uses the js most basic for loop implementation, for loop n times, the calculation time is about 0.2ms around.
Sure enough, the for loop is the most useful.
1000 times a second time consuming | 2000 times a second time consuming | time | |
---|---|---|---|
recursive | – | – | – |
Optimal recursive | 0.4 ms | 0.65 ms | O(n) |
The for loop | 0.2 ms | 0.22 ms | O(n) |
conclusion
So today we’ve seen three different ways to solve the Fibonacci sequence
-
recursive
Too many calls, too many performance problems for real use, but an understandable way to understand recursion.
-
Optimal recursive
Cache the recursive function that has been called, and the next time you use it, return the result of the recursive value directly. Good performance.
-
The for loop
Using the method of variable assignment, assign n to n-1, and n-1 to n-2, n times.
If anyone knows a way to do this faster than a for loop, let me know in the comments section below.