The classical example of a recursive function is the factorial function; we know that the factorial of 0 is 1, and we can define the factorial of an integer, say n, greater than one, as the number, n times the factorial of n-1, with the obvious c++ implementation as follows:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
unsigned factorial(unsigned n) | |
{ | |
if(n==0) | |
return 1; | |
else | |
return n*factorial(n-1); | |
} | |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
unsigned fibo(unsigned n) | |
{ | |
if(n<=1) | |
return n; | |
return fibo(n-1)+fibo(n-2); | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
unsigned fibo_memo(unsigned n) | |
{ | |
static map<unsigned,unsigned> memo; | |
if(n<=1) | |
return n; | |
if(memo.count(n)>0) | |
return memo[n]; | |
// otherwise | |
unsigned ret=fibo_memo(n-1)+fibo_memo(n-2); | |
memo[n]=ret; | |
return ret; | |
} |
This code is, in practice, much much faster than the other one (goes from exponential, to n*log(n) ). This general technique of storing already-calculated values for a function in a cache is called memoization.
We can generalize this pattern, and create a generic memoizer function, called memoize. Since we don't know the type of the input or output values, we need a templatized function, with two type parameters, as follows:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
template<class InType, class OutType> | |
std::function<OutType(InType)> memoize(std::function<OutType(InType)> foo) | |
{ | |
// return a lambda, this is a function | |
return [foo](InType n){ | |
static map<InType,OutType> memo; | |
OutType ret; | |
if(memo.count(n)>0) { | |
ret =memo[n]; | |
return ret; | |
} | |
ret=foo(n); | |
memo[n]=ret; | |
return ret; | |
}; | |
} |
To use it, we would need to define our function with a lambda:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
std::function<unsigned(unsigned)> fibo1=[](unsigned n) | |
{ | |
if(n<=1) | |
return n; | |
return fibo1(n-1)+fibo1(n-2); | |
}; | |
And in other place, replace the variable with its memoized variation:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
int main(int argc, char*argv[]) | |
{ | |
fibo1=memoize(fibo1); | |
cout << argv[1] << " ->" << fibo1(atoi(argv[1])) << endl; | |
} |
Of course, this would only work for functions that take only one argument, but it would not be hard to do it for other kinds of functions, although the pattern is more important than the particular functions :).
Many would realize that there's a better iterative algorithm for Fibonacci numbers; problems where memoizing helps can be solved with dynamic programming iterative algorithms, but, in general need a better understanding of the algorithm; when faced with a new problem, we can usually start with a recursive solution, memoize if we have performance problems and we suspect or know we're solving the same subproblems repeatedly, and move to dynamic programming only if we cannot achieve good performance with memoization.