【问题标题】:Stack overflow with Fibonacci function斐波那契函数的堆栈溢出
【发布时间】:2016-01-03 15:23:50
【问题描述】:

我还在学习 F#,所以请对我温柔一点!我试图定义一个函数来生成这样的斐波那契数......

let rec fib n =
  match n with
  | 0 -> 1
  | 1 -> 1
  | n -> fib(n-1) + (fib n-2)

然而,虽然这给出了 0 和 1 的正确结果,但它给出了 2 的堆栈溢出。我知道这不是尾递归,但对于 2 的输入,我不认为这是一个问题.

我认为模式匹配一​​直在进行,因此输入 2 时,它将匹配第三个模式,结果为 1+1(即 fib 0 和 fib 1 的结果)。

为什么我会收到 SO?

【问题讨论】:

  • 你的括号错了。现在它相当于| n -> fib(n-1) + ((fib n) - 2),它调用fib n,直到堆栈溢出。应该是| n -> fib (n-1) + fib (n-2)。这是一个简单的错字,所以我投票结束了这个问题。
  • @JakubLortz 感谢您对我的错误的解释。不知道你为什么要结束这个问题,我敢肯定我不是第一个(或最后一个)犯这个错误的人。可能对其他人有用。
  • 附注:让你的函数尾递归以避免各种堆栈溢出
  • @Mikhail 是的,我知道,谢谢。我仍在学习 F#,并且一步一步地学习。我想要一个有效的 fib 函数,然后打算使用尾递归对其进行修改。不过感谢您的提醒!

标签: f# fibonacci


【解决方案1】:

你的声明:

| n -> fib(n-1) + (fib n-2)

意思:

| n -> fib(n-1) + fib(n)-2

如果我有这样的调用,它可以很好地与fib(n-1) 配合使用,因为每次调用 n 的值都会减 1,而对于fib(n),你会继续使用相同的值调用相同的方法,直到你得到堆栈溢出异常。为避免这种情况,您需要:

| n -> fib(n-1) + fib(n-2)

【讨论】:

  • 谢谢。我对 F# 很陌生,我试图在不需要的地方去掉括号,就像区分我的 F# 和 C# 一样。看来我在这种情况下有点过于热情了!
猜你喜欢
  • 2014-11-14
  • 2012-05-06
  • 2021-11-30
  • 2012-12-29
  • 2016-04-02
  • 1970-01-01
  • 2011-03-24
  • 2017-02-13
相关资源
最近更新 更多