题目描述
斐波那契数列:
F(1)=1,F(2)=1, F(n)=F(n-1)+F(n-2)(n>=3,n∈N*)
现在要求输入一个整数n,请你输出斐波那契数列的第n项(从0开始,第0项为0)。
解题思路
暴力法就不介绍了。最常用解法是动态规划,这里递推式已经直接给出,编程时注意存储子问题解即可。
普通版:
class Solution:
def Fibonacci(self, n):
# write code here
if n < 0:
return None
if n == 0 or n == 1:
return n
f = [0] * (n+1)
f[0], f[1] = 0, 1
for i in range(2, n+1):
f[i] = f[i-1] + f[i-2]
return f[-1]
时间复杂度O(n),空间复杂度O(n)。上述解法只能说基本合格,仔细分析存储空间,发现每个值用过一次后就不用了,因此一直保存就显得很浪费,所以可以从减少空间复杂度入手,优化代码。
优化版:
class Solution:
def Fibonacci(self, n):
# write code here
if n < 0:
return None
if n == 0 or n == 1:
return n
pre, cur = 0, 1
for i in range(2, n+1):
temp = pre + cur
pre = cur
cur = temp
return cur
时间复杂度O(n),空间复杂度O(1)。用pre,cur分别保存f(n-2),f(n-1)的值。实际上可以把temp变量也直接省掉,如下:
class Solution:
def Fibonacci(self, n):
# write code here
if n < 0:
return None
if n == 0 or n == 1:
return n
pre, cur = 0, 1
for i in range(2, n+1):
cur, pre = pre + cur, cur
return cur
要注意的是需要考虑负数输入,因为题中只说了n是整数,而斐波那契数列中要求n>=0。
拓展训练
题目一:青蛙跳台阶
牛客网 跳台阶
一只青蛙一次可以跳上1级台阶,也可以跳上2级。求该青蛙跳上一个n级的台阶总共有多少种跳法(先后次序不同算不同的结果)。
解题思路
和斐波那契数列一样,只是起始值略有不同。可以自己分析起始值。
实战
class Solution:
def jumpFloor(self, number):
# write code here
if number < 0:
return None
if number <= 2:
return number
pre, cur = 1, 2
for i in range(3, number+1):
cur, pre = cur + pre, cur
return cur
题目二:青蛙变态跳台阶
牛客网 变态跳台阶
一只青蛙一次可以跳上1级台阶,也可以跳上2级……它也可以跳上n级。求该青蛙跳上一个n级的台阶总共有多少种跳法。
解题思路
动态规划:
与前面题稍有不同,这里递推式如下:
f(n)=f(n-1)+f(n-2)+……f(1)
如何保存好等式右边累加值是关键。
class Solution:
def jumpFloorII(self, number):
# write code here
if number < 0:
return None
if number <= 1:
return number
pre_total = 1
for i in range(2, number+1):
cur = pre_total + 1
pre_total = cur + pre_total
return cur
很多题目跳出计算机思维,使用数学方式可以更快得到结论。
数学法一:
易知 :
f(n)=f(n-1)+f(n-2)+……f(1)
f(n-1)=f(n-2)+……f(1)
两式相减得f(n)=2f(n-1)。这种方法本质上还是动态规划,只不过利用数学方法使得递推式更简单了。代码如下:
class Solution:
def jumpFloorII(self, number):
# write code here
if number < 0:
return None
if number <= 1:
return number
cur = 1
for i in range(2, number+1):
cur *= 2
return cur
数学法二:
f(n) = 2n-1
使用数学归纳法证明,因为:
f(1) =1 = 20;
f(2)=2=21;
f(3)=4=22;
假设 f(n-1)=2n-2成立,因为f(n)=2f(n-1)(前面已证明),所以可得:
f(n) = 2n-1
class Solution:
def jumpFloorII(self, number):
# write code here
if number <= 0:
return 0
return 2**(number-1)
题目三:矩形覆盖
我们可以用21的小矩形横着或者竖着去覆盖更大的矩形。请问用n个21的小矩形无重叠地覆盖一个2*n的大矩形,总共有多少种方法?
解题思路
牛客网
实际上还是斐波那契数列,就看能不能从具体问题中抽象出来。
class Solution:
def rectCover(self, number):
# write code here
if number < 0:
return None
if number <= 2:
return number
cur, pre = 2, 1
for i in range(3, number+1):
cur, pre = cur + pre, cur
return cur