Просто чтобы продолжить ответ Джона, вот как вы можете запомнить функцию факториала, чтобы не пересчитывать все на каждом этапе:
public Func<T, TResult> Memoize<T, TResult>(Func<T, TResult> func)
{
Dictionary<T, TResult> _resultsCache = new Dictionary<T, TResult>();
return (arg) =>
{
TResult result;
if (!_resultsCache.TryGetValue(arg, out result))
{
result = func(arg);
_resultsCache.Add(arg, result);
}
return result;
};
}
...
Func<int, int> factorial = null; // Just so we can refer to it
factorial = x => x <= 1 ? 1 : x * factorial(x-1);
var factorialMemoized = Memoize(factorial);
var res = Enumerable.Range(1, 10).Select(x => factorialMemoized(x));
foreach (var outt in res)
Console.WriteLine(outt.ToString());
РЕДАКТИРОВАТЬ: на самом деле приведенный выше код неверен, потому что factorial
вызывает factorial
, а не factorialMemoized
. Вот лучшая версия:
Func<int, int> factorial = null; // Just so we can refer to it
Func<int, int> factorialMemoized = null;
factorial = x => x <= 1 ? 1 : x * factorialMemoized(x-1);
factorialMemoized = Memoize(factorial);
var res = Enumerable.Range(1, 10).Select(x => factorialMemoized(x));
foreach (var outt in res)
Console.WriteLine(outt.ToString());
С этим кодом factorial
вызывается 10 раз, против 55 раз для предыдущей версии