使用 lambda 运算符创建斐波那契数列
本文关键字:数列 lambda 运算符 创建 使用 | 更新日期: 2023-09-27 18:18:50
我正在尝试解决欧拉项目中的一个问题,该项目正在创建一个斐波那契级数列,直到 400 万,并添加该系列中的偶数,这显然是非常简单的任务,我在 2 分钟内回答了它,
int result=2;
int first=1;
int second=2;
int i=2;
while (i < 4000000)
{
i = first + second;
if (i % 2 == 0)
{
result += i;
}
first = second;
second = i;
}
Console.WriteLine(result);
但我想使用 lambda 表达式来做到这一点
我的努力是这样的
DelType del = (oldVal, newVal) =>((oldVal==0?1:newVal + newVal==1?2:oldVal+newVal) % 2 == 0) ? oldVal + newVal : 0;
int a=del(0, 1);
请建议如何完成此操作
我的第一个答案是完全误读了这个问题,但现在我已经重读了它(感谢MagnatLU!我建议这不太适合 lambda 表达式。但是,它非常适合迭代器块和 LINQ 的组合:
// Using long just to avoid having to change if we want a higher limit :)
public static IEnumerable<long> Fibonacci()
{
long current = 0;
long next = 1;
while (true)
{
yield return current;
long temp = next;
next = current + next;
current = temp;
}
}
...
long evenSum = Fibonacci().TakeWhile(x => x < 4000000L)
.Where(x => x % 2L == 0L)
.Sum();
使用此递归函数
Func<int, int> fib = null;
fib = (x) => x > 1 ? fib(x-1) + fib(x-2) : x;
示例用法:
Console.WriteLine(fib(10));
下面是作为lambda表达式的oneliner:
Func<int, int, int, int, int> fib = null;
fib = (n, a, b, res) => n == 0 ? res : fib(n - 1, b, a + b, n % 2 == 0 ? res : res + a + b);
// usage: fib(n, 1, 0, 0)
它在 x86 上使用 O(n( 堆栈空间和 O(n( 时间,在 x64 上使用 O(1( 堆栈空间(由于 x64 JIT 上的尾递归优化(,因此在 32 位系统上 n=400000 时它将失败。
编辑:它甚至从结尾开始计算系列元素,而不是开头,但你应该知道如何用 tailrec 将其作为 λ。
与 Func 双行类似,现在使用本地函数可以像这样完成:
int Fibonacci(int i) => i <= 1 ? i : Fibonacci(i - 1) + Fibonacci(i - 2);
using System;
using System.Collections;
using System.Collections.Generic;
using System.Linq;
public class Fibonacci : IEnumerable<int>{
delegate Tuple<int,int> update(Tuple<int,int> x);
update func = ( x ) => Tuple.Create(x.Item2, x.Item1 + x.Item2);
public IEnumerator<int> GetEnumerator(){
var x = Tuple.Create<int,int>(0,1);
while (true){
yield return x.Item1;
x = func(x);
}
}
IEnumerator IEnumerable.GetEnumerator() {
return GetEnumerator();
}
}
class Sample {
static public void Main(){
int result= (new Fibonacci()).TakeWhile(x => x < 4000000).Where(x => x % 2 == 0).Sum();
Console.WriteLine(result);//4613732
}
}
其他
public static class Seq<T>{
public delegate Tuple<T,T> update(Tuple<T,T> x);
static public IEnumerable<T> unfold(update func, Tuple<T,T> initValue){
var value = initValue;
while (true){
yield return value.Item1;
value = func(value);
}
}
}
class Sample {
static public void Main(){
var fib = Seq<int>.unfold( x => Tuple.Create<int,int>(x.Item2, x.Item1 + x.Item2), Tuple.Create<int,int>(0,1));
int result= fib.TakeWhile(x => x < 4000000).Where(x => x % 2 == 0).Sum();
Console.WriteLine(result);
}
}
如果你想要一个纯粹的递归lambda解决方案,看看这个答案,有几个链接,可以看到一些展示它是如何完成的文章的链接。
然而,那个超级的东西对我来说太疯狂了,所以我最好遵循另一个已经存在的答案。
你知道你可以做到:
Func<int,int,int> func = (first, second) => {
var result=2;
int i=2;
while (i < 4000000)
{
i = first + second;
if (i % 2 == 0)
{
result += i;
}
first = second;
second = i;
}
return result;
};
我知道
这是一个老问题,但我今天正在处理同样的问题,并得出了这个简洁的函数式解决方案,运行时间为 O(n(:
static int FibTotal(int limit, Func<int, bool> include, int last = 0, int current = 1)
{
if (current < limit)
return FibTotal(limit, include, current, last + current) +
(include(current) ? current : 0);
else
return 0;
}
如果你首先定义这个方便类,你也可以得到一个很好的单行解决方案(也许 .NET 框架中已经存在这样的东西,但我找不到它(:
public static class Sequence
{
public static IEnumerable<T> Generate<T>(T seed, Func<T, T> next)
{
while (true)
{
yield return seed;
seed = next(seed);
}
}
}
然后,解决方案变为:
var result = Sequence.Generate(Tuple.Create(1, 1),
t => Tuple.Create(t.Item2, t.Item1 + t.Item2))
.Select(t => t.Item1)
.TakeWhile(i => i < 4000000)
.Where(i=> i % 2 == 0)
.Sum();
public static void fibSeriesEx3()
{
List<int> lst = new List<int> { 0, 1 };
for (int i = 0; i <= 10; i++)
{
int num = lst.Skip(i).Sum();
lst.Add(num);
foreach (int number in lst)
Console.Write(number + " ");
Console.WriteLine();
}
}
一个有效的衬里:
Func<int, int, int, int, int> fib = null;
fib = (a, b, counter, n) => counter < n ? fib(b, a + b, counter+1, n) : a;
//print the 9th fib num
Console.WriteLine(fib(0,1,1,9));
输出: 21