每日一小练——高速Fibonacci数算法

上得厅堂,下得厨房,写得代码,翻得围墙,欢迎来到睿不可挡的每日一小练!

题目:高速Fibonacci数算法

内容:先说说Fibonacci数列,它的定义是数列:f1,f2....fn有例如以下规律:

尝试寻找高速的求出fn的方法

我的解法:上来没多想,打开vs2013就敲了起来,问题果然非常easy,分分钟就超神。。奥,不正确就攻克了!

事实上题目中就给出了这个算法的递归形式,所以首先我想到的是递归解法,只是由于求解高速方法在递归之前,我编写了一个非递归的算法

#include <iostream>
using namespace std;

int _tmain(int argc, _TCHAR* argv[])
{
	int f(int n);
	cout << f(7) << endl;
	getchar();
	return 0;
}

int f(int n)
{
	int temp = 0, f1 = 1, f2 = 1;
	if (n == 1 || n == 2)
		return 1;
	else
	{
		for (int i = 1; i < (n - 1); i++)
		{
			temp = f1 + f2;
			f2 = f1;
			f1 = temp;
		}
		return temp;
	}
}

然后我又编写了递归的算法

#include <iostream>
using namespace std;

int _tmain(int argc, _TCHAR* argv[])
{
	int f(int n);
	cout << f(7) << endl;
	getchar();
	return 0;
}

int f(int n)
{
	if (n == 1|| n==2)
		return 1;
	if (n > 2)
		return f(n - 1) + f(n - 2);
}

在递归的基础上,有人提出了更犀利的算法,这个我没有想到。。羞愧。。。

这个算法利用了一些技巧矩阵,通过矩阵乘法来算Fibonacci的加法,然后通过我在《数值自乘非递归解》中提到的利用区分奇偶数来利用指数二进制堆乘的方法,降低乘法的次数。

ps:

利用上面的矩阵连乘,在矩阵11位置的数就是矩阵11和21的和,而且用矩阵11和21表示Fibonacci的f(n-1)和f(n-2),通过连乘来求fn。

#include <iostream>
using namespace std;

int _tmain(int argc, _TCHAR* argv[])
{
	int f(int n);
	cout << f(7) << endl;
	getchar();
	return 0;
}

int f(int n)
{
	void matrix_power(int a, int b, int c, int d, int n, int *aa, int *bb, int *cc, int *dd);
	int a, b, c, d;
	if (n == 1 || n == 2)
	{
		return 1;
	}
	else
	{
		matrix_power(1, 1, 1, 0, n - 2, &a, &b, &c, &d);
		return a + b;
	}
}

void matrix_power(int a, int b, int c, int d, int n, int *aa, int *bb, int *cc, int *dd)
{
	int xa, xb, xc, xd;
	if (n == 1)
		*aa = a, *bb = b, *cc = c, *dd = d;
	else if (n & 0x01 == 1)
	{
		matrix_power(a, b, c, d, n - 1, &xa, &xb, &xc, &xd);
		*aa = a*xa + b*xc;
		*bb = a*xb + b*xd;
		*cc = c*xa + d*xc;
		*dd = c*xb + d*xd;
	}
	else
	{
		matrix_power(a, b, c, d, n >> 1, &xa, &xb, &xc, &xd);
		*aa = xa*xa + xb*xc;
		*bb = xa*xb + xb*xd;
		*cc = xc*xa + xd*xc;
		*dd = xc*xb + xd*xd;
	}
}

三段代码的实验结果同例如以下:

欢迎大家增加每日一小练,嘿嘿!

每天练一练,日久见功夫,加油!

      -End-

參考文献:《c语言名题精选百则》

时间: 2024-10-21 12:35:45

每日一小练——高速Fibonacci数算法的相关文章

每日一小练——快速Fibonacci数算法

上得厅堂,下得厨房,写得代码,翻得围墙,欢迎来到睿不可挡的每日一小练! 题目:快速Fibonacci数算法 内容:先说说Fibonacci数列,它的定义是数列:f1,f2....fn有如下规律: 尝试寻找快速的求出fn的方法 我的解法:上来没多想,打开vs2013就敲了起来,问题果然很简单,分分钟就超神..奥,不对就解决了! 其实题目中就给出了这个算法的递归形式,所以首先我想到的是递归解法,不过因为求解快速方法在递归之前,我编写了一个非递归的算法 #include <iostream> usi

每日一小练——列出全部子集

上得厅堂,下得厨房,写得代码,翻得围墙,欢迎来到睿不可挡的每日一小练! 题目:列出全部子集 内容:写一个程序,列出{1,2,3,4,...,n}这个集合的全部子集,包含空集{ }. 解决问题的算法应该有非常多种,只是我发现了一种非常easy也非常好玩的方法,就是用二进制表示几何元素的方法 比方假设n是3,则子集有(不包含空集){1}{2}{3}{1,2}{1,3}{2,3}{1,2,3}就是1*{0,1}and2*{0,1}and3{0,1}每一个组合出现一次,所以也能够看成是{0,0,1}{0

每日一小练——Armstrong数

上得厅堂,下得厨房,写得代码,翻得围墙,欢迎来到睿不可挡的每日一小练! 题目:Armstrong数 内容: 在三位的正整数中,例如abc,有一些可以满足a^3+b^3+c^3=abc的条件,也就是说各个位数的立方和正好是该数本身,这些数称为Armstrong数.是编写一个程序求出所有的三位Armstrong数. 我的解法:上来没多想,打开vs2013就敲了起来,问题果然很简单,分分钟就超神..奥,不对就解决了!Armstrong数在国内好像就叫做水仙花数,很多程序设计的课本后面都有这种类型的思考

每日一小练——列出所有子集

上得厅堂,下得厨房,写得代码,翻得围墙,欢迎来到睿不可挡的每日一小练! 题目:列出所有子集 内容:写一个程序,列出{1,2,3,4,...,n}这个集合的所有子集,包括空集{ }. 解决这个问题的算法应该有很多种,不过我发现了一种很简单也很好玩的方法,就是用二进制表示几何元素的方法 比如如果n是3,则子集有(不包括空集){1}{2}{3}{1,2}{1,3}{2,3}{1,2,3}就是1*{0,1}and2*{0,1}and3{0,1}每个组合出现一次,所以也可以看成是{0,0,1}{0,1,0

每日一小练——Eratosthenes 筛选法

上得厅堂.下得厨房.写得代码,翻得围墙,欢迎来到睿不可挡的每日一小练! 题目:Eratosthenes筛选法 内容: 求质数是一个非常普遍的问题,通常不外乎用数去除.除到不尽时,给定的数就是质数. 可是早在2000年前人们就知道了一个不必用除法而找出2~N的全部质数的方法.如果一个非常奇妙的筛子,能够给出一个数.比如i,这个筛子有办法把i全部的倍数去掉. 请用这种方法求出2~N之间的全部质数.即Eratosthenes筛选法. 我的解法:上来没多想.打开vs2013就敲了起来.问题果然非常eas

每日一小练——最长平台问题

上得厅堂,下得厨房,写得代码,翻得围墙,欢迎来到睿不可挡的每日一小练! 题目:最长平台问题 内容:一直一个已经从小到大排序的数组,这个数组中的一个平台就是连续的一串同样的元素.而且这个元素不能再延伸. 比如,在1,2,2,3,3,3,4,5,5,6中1,2,2,3,3,3,4,5,5,6都是平台.试编写一个程序,接受一个数组,把这个数组中最长的平台找出来.在这个样例中, 3,3,3就是该数组的中的最长的平台. 说明: 这个程序十分简单,可是编写好却不easy,因此在编敲代码时应注意考虑以下几点:

每日一小练——求质数

上得厅堂,下得厨房,写得代码,翻得围墙,欢迎来到睿不可挡的每日一小练! 题目:求质数 内容: 试编写一个程序,找出前N个质数.如果没有进一步要求,这不是难题.但在此希望从所知的.使用除法的方法中,用最快的办法来编写程序. 我的解法:上来没多想,打开vs2013就敲了起来,问题果然很简单,分分钟就超神..奥,不对就解决了!这个题目确实很简单,先看看常规解法吧! #include <iostream> #include <math.h> #define endNum 200 using

每日一小练——等值首尾和

上得厅堂,下得厨房,写得代码,翻得围墙,欢迎来到睿不可挡的每日一小练! 题目:等值首尾和 内容: 假设有一个数组x[],它有n个元素,每一个都大于零:称x[0]+x[1]+...+x[i]为前置和,而x[j]+x[j+1]+...+x[n-1]为后置和.试编写一个程序,求出x[]中有多少组相同的前置和后置和. 例如:x[]的元素是3,6,2,1,4,5,2,于是x[]的前置和有以下7个,即3,9,11,12,16,21,23:后置和则2,7,11,12,14,20,23; 于是11,12,23,

每日一小练——数值自乘非递归解

上得厅堂,下得厨房,写得代码,翻得围墙,欢迎来到睿不可挡的每日一小练! 题目:数值自乘非递归解 内容: 连续求m^n问题(m与n是正整数).前面的提示会得到一个递归程序,请编写一个运算效率同样高的非递归的程序. 我的解法:上来没多想,打开vs2013就敲了起来,问题果然很简单,分分钟就超神..奥,不对就解决了!如果是非递归其实一种简单的方法就是把, 递归的几种分类做好if分支,用一个循环处理就好了,不过如果我们认真思考m^n这个式子,我们会发现n如果用二进制表示的话如:2^7可改写为 2^011