I am making a program for nth Fibonacci number. I made the following program using recursion and memoization. The main problem is that the value of n can go up to 10000 which means that the Fibonacci number of 10000 would be more than 2000 digit long.
With a little bit of googling, I found that i could use arrays and store every digit of the solution in an element of the array but I am still not able to figure out how to implement this approach with my program.
#include<iostream>
using namespace std;
long long int memo[101000];
long long int n;
long long int fib(long long int n)
{
if(n==1 || n==2)
return 1;
if(memo[n]!=0)
return memo[n];
return memo[n] = fib(n-1) + fib(n-2);
}
int main()
{
cin>>n;
long long int ans = fib(n);
cout<<ans;
}
How do I implement that approach or if there is another method that can be used to achieve such large values?
Try not to use recursion for a simple problem like fibonacci. And if you'll only use it once, don't use an array to store all results. An array of 2 elements containing the 2 previous fibonacci numbers will be enough. In each step, you then only have to sum up those 2 numbers. How can you save 2 consecutive fibonacci numbers? Well, you know that when you have 2 consecutive integers one is even and one is odd. So you can use that property to know where to get/place a fibonacci number: for
fib(i)
, ifi
is even (i%2
is 0) place it in the first element of the array (index 0), else (i%2
is then 1) place it in the second element(index 1). Why can you just place it there? Well when you're calculatingfib(i)
, the value that is on the placefib(i)
should go isfib(i-2)
(because(i-2)%2
is the same asi%2
). But you won't needfib(i-2)
any more:fib(i+1)
only needsfib(i-1)
(that's still in the array) andfib(i)
(that just got inserted in the array).So you could replace the recursion calls with a
for loop
like this:Recursion is actually discommanded while programming because of the time(function calls) and ressources(stack) it consumes. So each time you use recursion, try to replace it with a loop and a stack with simple pop/push operations if needed to save the "current position" (in c++ one can use a
vector
). In the case of the fibonacci, the stack isn't even needed but if you are iterating over a tree datastructure for example you'll need a stack (depends on the implementation though). As I was looking for my solution, I saw @naomik provided a solution with thewhile
loop. That one is fine too, but I prefer the array with the modulo operation (a bit shorter).Now concerning the problem of the size
long long int
has, it can be solved by using external libraries that implement operations for big numbers (like the GMP library or Boost.multiprecision). But you could also create your own version of aBigInteger
-like class from Java and implement the basic operations like the one I have. I've only implemented the addition in my example (try to implement the others they are quite similar).The main idea is simple, a
BigInt
represents a big decimal number by cutting its little endian representation into pieces (I'll explain why little endian at the end). The length of those pieces depends on the base you choose. If you want to work with decimal representations, it will only work if your base is a power of 10: if you choose 10 as base each piece will represent one digit, if you choose 100 (= 10^2) as base each piece will represent two consecutive digits starting from the end(see little endian), if you choose 1000 as base (10^3) each piece will represent three consecutive digits, ... and so on. Let's say that you have base 100, 12765 will then be[65, 27, 1]
, 1789 will be[89, 17]
, 505 will be[5, 5]
(= [05,5]), ... with base 1000: 12765 would be[765, 12]
, 1789 would be[789, 1]
, 505 would be[505]
. It's not the most efficient, but it is the most intuitive (I think ...)The addition is then a bit like the addition on paper we learned at school:
BigInt
BigInt
(if it has pieces left)For example:
Here is a demo where I used the class above to calculate the fibonacci of 10000 (result is too big to copy here)
Good luck!
PS: Why little endian? For the ease of the implementation: it allows to use
push_back
when adding digits and iteration while implementing the operations will start from the first piece instead of the last piece in the array.One thing that I think should be pointed out is there's other ways to implement
fib
that are much easier for something like C++ to computeconsider the following pseudo code
This doesn't require memoisation and you don't have to be concerned about blowing up your stack with too many recursive calls. Recursion is a really powerful looping construct but it's one of those fubu things that's best left to langs like Lisp, Scheme, Kotlin, Lua (and a few others) that support it so elegantly.
That's not to say tail call elimination is impossible in C++, but unless you're doing something to optimise/compile for it explicitly, I'm doubtful that whatever compiler you're using would support it by default.
As for computing the exceptionally large numbers, you'll have to either get creative doing adding The Hard Way or rely upon an arbitrary precision arithmetic library like GMP. I'm sure there's other libs for this too.
Adding The Hard Way™
Remember how you used to add big numbers when you were a little tater tot, fresh off the aluminum foil?
5-year-old math
Well you gotta add each column, right to left. And when a column overflows into the double digits, remember to carry that 1 over to the next column.
The 10,000th fibonacci number is thousands of digits long, so there's no way that's going to fit in any integer C++ provides out of the box. So without relying upon a library, you could use a string or an array of single-digit numbers. To output the final number, you'll have to convert it to a string tho.
(woflram alpha:
fibonacci 10000
)Doing it this way, you'll perform a couple million single-digit additions; it might take a while, but it should be a breeze for any modern computer to handle. Time to get to work !
Here's an example in of a
Bignum
module in JavaScriptWe can verify that the Wolfram Alpha answer above is correct
Expand the program below to run it in your browser