算法 斐波那契数列
斐波那契数列: f(n)=f(n-1)+f(n-2); n>=2
f(0)=0; f(1)=1;
即有名的兔子繁衍问题。
斐波那契数列共有三种解法,总结一下。

第一种:递归求解

递归求解比较简单,是大家常见的一种解法。

  1. int fibonacci(int n)
  2. {
  3. cout<<"calculating "<<n<<endl;
  4. if (n<=0) {
  5. return 0;
  6. }
  7. if (n==1) {
  8. return 1;
  9. }
  10. return fb(n-1)+fb(n-2);
  11. }

关于这种解法,不再赘述,下面主要说下时间复杂度分析。
设f(n)为参数为n时的时间复杂度,很明显:f(n)=f(n-1)+f(n-2)
这就转化为了数学上的二阶常系数差分方程,并且为其次方程。
即转化为了求f(n)的值,f(n)=f(n-1)+f(n-2)且f(0)=0; f(1)=1;
特征方程为:x^2-x-1=0
得 x=(1±√5)/2
因而f(n)的通解为:
关于斐波那契数列三种解法及时间复杂度分析 - 图1
f(0)=0; f(1)=1可解得c_1,c_2
最终可得,时间复杂度为:
关于斐波那契数列三种解法及时间复杂度分析 - 图2

第一种解法比较简单,但是多个元素重复计算,因而时间复杂度较高,为了避免重复计算,可进行循环计算减少时间复杂度.

第二种:时间复杂度为O(n)

  1. int Fibonacci(int n) {
  2. if (n<=0) {
  3. return 0;
  4. }
  5. if (n==1) {
  6. return 1;
  7. }
  8. int min=0;
  9. int max=1;
  10. int i=2;
  11. int result=0;
  12. while (i<=n) {
  13. result=min+max;
  14. min=max;
  15. max=result;
  16. ++i;
  17. }
  18. return result;
  19. }

第三种:还有一种时间复杂度更低的算法。

根据上面的递归公式,可以得到
关于斐波那契数列三种解法及时间复杂度分析 - 图3
因而计算f(n)就简化为了计算矩阵的(n-2)次方,而计算矩阵的(n-2)次方,又可以进行分解,即计算矩阵(n-2)/2次方的平方,逐步分解下去,由于折半计算矩阵次方,因而时间复杂度为O(log n) 。
具体代码实现如下:

  1. //
  2. // main.cpp
  3. // fibonaccimatrix
  4. //
  5. // Created by shunagao on 15/8/31.
  6. // Copyright © 2015年 shunagao. All rights reserved.
  7. //
  8. #include <iostream>
  9. using namespace std;
  10. class Matrix
  11. {
  12. public:
  13. int n;
  14. int **m;
  15. Matrix(int num)
  16. {
  17. m=new int*[num];
  18. for (int i=0; i<num; i++) {
  19. m[i]=new int[num];
  20. }
  21. n=num;
  22. clear();
  23. }
  24. void clear()
  25. {
  26. for (int i=0; i<n; ++i) {
  27. for (int j=0; j<n; ++j) {
  28. m[i][j]=0;
  29. }
  30. }
  31. }
  32. void unit()
  33. {
  34. clear();
  35. for (int i=0; i<n; ++i) {
  36. m[i][i]=1;
  37. }
  38. }
  39. Matrix operator=(const Matrix mtx)
  40. {
  41. Matrix(mtx.n);
  42. for (int i=0; i<mtx.n; ++i) {
  43. for (int j=0; j<mtx.n; ++j) {
  44. m[i][j]=mtx.m[i][j];
  45. }
  46. }
  47. return *this;
  48. }
  49. Matrix operator*(const Matrix &mtx)
  50. {
  51. Matrix result(mtx.n);
  52. result.clear();
  53. for (int i=0; i<mtx.n; ++i) {
  54. for (int j=0; j<mtx.n; ++j) {
  55. for (int k=0; k<mtx.n; ++k) {
  56. result.m[i][j]+=m[i][k]*mtx.m[k][j];
  57. }
  58. }
  59. }
  60. return result;
  61. }
  62. };
  63. int main(int argc, const char * argv[]) {
  64. unsigned int num=2;
  65. Matrix first(num);
  66. first.m[0][0]=1;
  67. first.m[0][1]=1;
  68. first.m[1][0]=1;
  69. first.m[1][1]=0;
  70. int t;
  71. cin>>t;
  72. Matrix result(num);
  73. result.unit();
  74. int n=t-2;
  75. while (n) {
  76. if (n%2) {
  77. result=result*first;
  78. }
  79. first=first*first;
  80. n=n/2;
  81. }
  82. cout<<(result.m[0][0]+result.m[0][1])<<endl;
  83. return 0;
  84. }