第2章 数字之魅——斐波那契(Fibonacci)数列
斐波那契(Fibonacci)数列
问题描述
递归算法:
package chapter2shuzizhimei.fibonacci;
/**
* Fibonacci数列递归求解
* @author DELL
*
*/
public class Fibonacci1 {
public static int fibonacci(int n){
if(n<=0)
return 0;
else if(n==1)
return 1;
else
return fibonacci(n-1)+fibonacci(n-2);
}
public static void main(String[] args) {
int n = 3;
System.out.println("fibonacci("+n+") = "+fibonacci(n)); } }
我们的问题是:有没有更加优化的解法?
分析与解法
【解法一】递推关系的优化
上述递归算法中有着很多的重复计算,利用一个数组存储中间结果避免重复计算。时间复杂度为O(N),空间复杂度也为O(N)。
算法如下:
package chapter2shuzizhimei.fibonacci;
/**
* Fibonacci数列求解
* 【解法一】递推关系的优化
* @author DELL
*
*/
public class Fibonacci2 {
private static int f[];
public Fibonacci2(int n){
f = new int[n+1];
for(int i=0;i<n;i++){
f[i]=-1;
}
}
public static int fibonacci(int n){
if(n<=0){
f[0]=0;
return 0;
}
else if(n==1){
f[1]=1;
return 1;
}
else{
if(f[n-1]==-1){
if(f[n-2]==-1)
return fibonacci(n-1)+fibonacci(n-2);
else
return fibonacci(n-1)+f[n-2];
}else{
return f[n-1]+f[n-2];
}
}
}
public static void main(String[] args) {
int n = 3;
new Fibonacci2(n);
System.out.println("fibonacci("+n+") = "+fibonacci(n)); } }
程序运行结果如下:
fibonacci(3) = 2
【解法二】采用非递归求解
算法如下:
package chapter2shuzizhimei.fibonacci;
/**
* Fibonacci数列求解
* 【解法二】非递归
* @author DELL
*
*/
public class Fibonacci3 {
public static int fibonacci(int n){
if(n<=0)
return 0;
else if(n==1)
return 1;
else{
int f0 = 0,f1 = 1,f2 = 0;
for(int i=2;i<=n;i++){
f2 = f0 + f1;
f0 = f1;
f1 = f2;
}
return f2;
}
}
public static void main(String[] args) {
int n = 3;
System.out.println("fibonacci("+n+") = "+fibonacci(n)); } }
程序运行结果如下:
fibonacci(3) = 2
【解法三】求解通项公式


算法代码如下:
package chapter2shuzizhimei.fibonacci;
/**
* Fibonacci数列求解
* 【解法三】求解通项公式
* @author DELL
*
*/
public class Fibonacci4 {
public static long fibonacci(int n){
double x = Math.sqrt(5);
double f = (x/5)*Math.pow((1+x)/2, n) - (x/5)*Math.pow((1-x)/2, n);
return Math.round(f);
}
public static void main(String[] args) {
int n = 3;
System.out.println("fibonacci("+n+") = "+fibonacci(n)); } }
程序运行结果如下:
fibonacci(3) = 2
【解法四】分治策略


要先导入JAMA:Java矩阵包
package chapter2shuzizhimei.fibonacci; import Jama.Matrix; /**
* Fibonacci数列求解
* 【解法四】分治策略
* @author DELL
*
*/
public class Fibonacci5 {
//求解矩阵A的n次方
public static Matrix MatrixPow(Matrix A, int n){
int m = A.getColumnDimension();
Matrix result = new Matrix(m,m); //生成全为0的矩阵
for(int i=0;i<m;i++){ //变成单位矩阵
result.set(i, i, 1);
}
Matrix temp = A;
while(n!=0){
if((n&0x01)==1)
result = result.times(temp); //矩阵的乘法
temp = temp.times(temp);
n >>= 1;
}
return result;
}
//计算Fibonacci数列
public static long fibonacci(int n){
if(n<=0)
return 0;
else if(n==1)
return 1;
else{
Matrix A = new Matrix(2,2,1); //生成全为1的矩阵
A.set(1, 1, 0);
Matrix B = MatrixPow(A, n-1);
return (long) B.get(0, 0);
}
}
public static void main(String[] args) {
int n = 5;
System.out.println("fibonacci("+n+") = "+fibonacci(n)); } }
程序运行结果如下:
fibonacci(5) = 5
扩展问题
假设A(0)=1,A(1)=2,A(2)=2。对于n>2,都有A(K) = A(k-1) + A(k-2) +A(k-3)。
1. 对于任何一个给定的n,如何计算出A(n)?
2. 对于n非常大的情况,如n=260的时候,如何计算A(n) mod M (M<100000)呢?
问题1:非递归解法,代码如下:
package chapter2shuzizhimei.fibonacci;
/**
* 扩展问题1求解
* 非递归
* @author DELL
*
*/
public class Fibonacci6 {
public static int A(int n){
if(n<=0)
return 1;
else if(n==1||n==2)
return 2;
else{
int f0 = 1,f1 = 2,f2 = 2,f3 = 0;
for(int i=3;i<=n;i++){
f3 = f0 + f1 + f2;
f0 = f1;
f1 = f2;
f2 = f3;
}
return f3;
}
}
public static void main(String[] args) {
int n = 4;
System.out.println("A("+n+") = "+A(n)); } }
程序运行结果如下:
A(4) = 9
问题2:非递归解法,代码如下:
package chapter2shuzizhimei.fibonacci;
/**
* 扩展问题2求解
* 非递归
* @author DELL
*
*/
public class Fibonacci7 {
//计算A(n) mod m
public static long A(long n,long m){
if(n<=0)
return 1;
else if(n==1||n==2)
return 2;
else{
long f0 = 1,f1 = 2,f2 = 2,f3 = 0;
for(int i=3;i<=n;i++){
f0 = f0%m;
f1 = f1%m;
f2 = f2%m;
f3 = f0 + f1 + f2;
f0 = f1;
f1 = f2;
f2 = f3;
}
return f3%m;
}
}
public static void main(String[] args) {
long n = (long) Math.pow(2, 10);
long m = 100;
System.out.println("A("+n+") = "+A(n,m)); } }
程序运行结果如下:
A(1024) = 97
第2章 数字之魅——斐波那契(Fibonacci)数列的更多相关文章
- 斐波那契(Fibonacci)数列的几种计算机解法
题目:斐波那契数列,又称黄金分割数列(F(n+1)/F(n)的极限是1:1.618,即黄金分割率),指的是这样一个数列:0.1.1.2.3.5.8.13.21.34.…….在数学上,斐波纳契数列以如下 ...
- 斐波那契(Fibonacci)数列的七种实现方法
废话不多说,直接上代码 #include "stdio.h" #include "queue" #include "math.h" usin ...
- 如何用Python输出一个斐波那契Fibonacci数列
a,b = 0, 1 while b<100: print (b), a, b = b, a+b
- 斐波那契 (Fibonacci)数列
尾递归会将本次方法的结果计算出来,直接传递给下个方法.效率很快. 一般的递归,在本次方法结果还没出来的时候,就调用了下次的递归, 而程序就要将部分的结果保存在内存中,直到后面的方法结束,再返回来计算. ...
- ACM/ICPC 之 数论-斐波拉契●卢卡斯数列(HNNUOJ 11589)
看到这个标题,貌似很高大上的样子= =,其实这个也是大家熟悉的东西,先给大家科普一下斐波拉契数列. 斐波拉契数列 又称黄金分割数列,指的是这样一个数列:0.1.1.2.3.5.8.13.21.34.… ...
- 2019.8.3 [HZOI]NOIP模拟测试12 A. 斐波那契(fibonacci)
2019.8.3 [HZOI]NOIP模拟测试12 A. 斐波那契(fibonacci) 全场比赛题解:https://pan.baidu.com/s/1eSAMuXk 找规律 找两个节点的lca,需 ...
- 递归函数练习:输出菲波拉契(Fibonacci)数列的前N项数据
/*====================================================================== 著名的菲波拉契(Fibonacci)数列,其第一项为0 ...
- [洛谷P3938]:斐波那契(fibonacci)(数学)
题目传送门 题目描述 小$C$养了一些很可爱的兔子.有一天,小$C$突然发现兔子们都是严格按照伟大的数学家斐波那契提出的模型来进行繁衍:一对兔子从出生后第二个月起,每个月刚开始的时候都会产下一对小兔子 ...
- HZOJ 斐波那契(fibonacci)
先说一个规律: 如图将每个月出生的兔子的编号写出来,可以发现一只兔子在哪一列他的父亲就是谁. 每列的首项可以通过菲波那契求得. 然后你就可以像我一样通过这个规律打表每个点的父亲,预处理出倍增数组,倍增 ...
随机推荐
- POJ 3744 Scout YYF I (概率dp+矩阵快速幂)
题意: 一条路上,给出n地雷的位置,人起始位置在1,向前走一步的概率p,走两步的概率1-p,踩到地雷就死了,求安全通过这条路的概率. 分析: 如果不考虑地雷的情况,dp[i],表示到达i位置的概率,d ...
- [转]关于GCD与多线程
GCD是什么,你知道吗?你知道了GCD,你确定你会使用吗? 这一篇文章是站在初学者角度去分析GCD,原因是这个很多iOS开发者根本就没用过,即使用过,不知道其中的原理.讲解之前认识一下什么是线程,为什 ...
- 【原】Mongodb相关资料
Mongodb与关系型数据库对比 Mongodb与关系型数据库对比 由于之前主要接触的是关系型数据库,所以主要将Mongodb与关系型数据库进行对比:主要从术语.Server与Client.数据定义语 ...
- Codeforces Round #365 (Div. 2) C - Chris and Road 二分找切点
// Codeforces Round #365 (Div. 2) // C - Chris and Road 二分找切点 // 题意:给你一个凸边行,凸边行有个初始的速度往左走,人有最大速度,可以停 ...
- SRM 514 DIV1 500pt(DP)
题目简述 给定一个H×W大小的矩阵,每个格子要么是1~9中的一个数,要么是".",要求你把“.”填成具体的数字(1~9),并且符合以下两个要求: 对于所有的整数r 和 c( 0 & ...
- (转)UML实践----用例图、类图、对象图、顺序图、协作图、状态图、活动图、组件图、配置图
面向对象的问题的处理的关键是建模问题.建模可以把在复杂世界的许多重要的细节给抽象出.许多建模工具封装了UML(也就是Unified Modeling Language™),这篇课程的目的是展示出UML ...
- 咏南DATASNAP中间件提供免费使用了
咏南DATASNAP中间件提供免费使用了. 百度网盘分享: 链接: http://pan.baidu.com/s/1c06Ivp2 密码: dhhm
- [iOS UI进阶 - 1] 自定义控件
A.关于Quiartz2D的一些细节 1.UIKit的工具已经封装了上下文引用,所以不用手动获取和渲染 - (void)drawRect:(CGRect)rect { [[UIColor redCol ...
- javascript中document对象的属性和方法
document.documentElement; document.firstChild;document.childNodes[0];// 取得对<html>的引用document.b ...
- iOS异常捕获
文章目录 一. 系统Crash 二. 处理signal 下面是一些信号说明 关键点注意 三. 实战 四. Crash Callstack分析 – 进⼀一步分析 五. demo地址 六. 参考文献 前言 ...