首先让我们来了解一下什么是递归,它有什么条件。
程序调用自身的编程技巧称为递归( recursion)。 递归做为一种算法在程序设计语言中广泛应用。 一个过程或函数在其定义或说明中有直接或间接调用自身的一种方法,它通常把一个大型复杂的问题层层转化为一个与原问题相似的规模较小的问题来求解,递归策略只需少量的程序就可描述出解题过程所需要的多次重复计算,大大地减少了程序的代码量。 递归的主要思考方式在于:把大事化小
递归的两个必要条件:
在写递归函数时需要切记这两个条件。
递归也要考虑栈溢出等问题,比如斐波那契数列数列的递归实现。
传说婆罗门庙里有一个塔台,台上有 3 根标号为 A,B,C的用钻石做
成的柱子,在 A 柱上放着 64 个金盘,每一个都比下面的略小一点。把 A 柱上的金盘全部移到C柱上的那一天就是世界末日。移动的条件是:一次只能移动一个金盘,移动过程中大金盘不能放在小金盘上面。庙里的僧人一直在移个不停。因为全部的移动是2^63次,如果每秒移动一次的话,需要 500 亿年。
设最初的盘子总数为n
当n=3时:
看一下示例图
利用这个解法,
将移动n 个盘子的汉诺塔问题归结为移动(n-1)个盘子的汉诺塔问题。与此类似,移动(n-1)个盘子的汉诺塔问题又可归结为移动(n-2)个盘子的汉诺塔问题……最后总可以归结到只移动一个盘子的汉诺塔问题,这样问题就解决。
#include
void Hanoi(int n, char a ,char b ,char c)
{
if (n == 1)
{
printf("%c->%c\n", a , c);//只有一个盘子,直接移动到C
}
else
{
Hanoi(n - 1,a ,c ,b );//将A上面n-1个盘子移动到B中
printf("%c->%c\n", a ,c);//将A最后一个移动到C
Hanoi(n - 1, b, a, c);//将B上的n-1个盘子移到C上
}
}
int main()
{
char a = 'A';
char b = 'B';
char c = 'C';
Hanoi(3,a,b,c);
return 0;
}
暂不考虑特别大的数使栈溢出。
对于青蛙跳台阶,由于步骤较多,难以画图,我们采用数学推理的方法,求得相关递推公式。
当n = 1,只有一种跳法,那就是1
当n = 2,那么有两种跳法,2,[1, 1]
当n = 3,那么有三种跳法,[1, 1, 1], , [1, 2], [2, 1]
当n = 4,那么有五种跳法,[1, 1, 1, 1], [1, 1, 2], [1, 2, 1], [2, 1, 1], [2, 2]
当n = 5,那么有八种跳法,[1, 1, 1, 1, 1], [1, 1, 1, 2], [1, 1, 2, 1], [1, 2, 1, 1], [2, 1, 1, 1], [2, 2, 1], [2, 1, 2], [1, 2, 2]
结果为1,2,3,5,8
有没有感觉很眼熟
没错,这就是一个斐波那契数列,所以问题很快可以解决。
//递归做法:
int RJump1(int n)
{
if (n == 0)
return 0;
if (n == 1)
return 1;
if (n == 2)
return 2;
//n>2
return RJump1(n - 1) + RJump1(n - 2);
}
//非递归做法:
int Jump1(int n)
{
if (n == 0)
return 0;
if (n == 1)
return 1;
if (n == 2)
return 2;
int a1 = 1;
int a2 = 2;
int a3;
for (int i = 3; i <= n; i++)
{
a3 = a1 + a2;
a1 = a2;
a2 = a3;
}
return a3;
}
如果台阶级数为n的话,这时我们把n级台阶时的跳法看成n的函数,记函数关系为F,第一次跳的时候有n种不同的选择:
第三种情况比较难以想到,我也是想了很久才想明白。
所以当台阶为n时总次数为
F(n)=F(n-1)+F(n-2)……+F(1)+F(0)
令n=n-1
此时最多第一次跳n-1次
F(n-1)=F(n-2)……+F(0)
两式相减即可得到
F(n)=2*F(n-1)
我们得到了递推公式,以及F(1)=F(0)=1。
但是当n=1和n=0时不满足递推公式,所以当n=1或n=0时返回1即可。
跳法为1,1,2,4,8……
其实可以由递推公式看出是一个等比数列,可以算出它的和2^n-1。
int RJump2(int n)
{
//因为递推公式只需要找到它前一项,即F(n-1)的值就可以返回,我们知道n=1和n=0是1种,是已经确定的值,到n=1或n=0即可结束
if (n == 1||n == 0)
return 1;
//当阶数大于1时可以用递推公式
return 2 * RJump2(n - 1);
}
直接使用公式即可。代码简单得让人害怕。
同样我们采用问题2的方法,使用函数推理
当m
F(n)=F(n-1)+F(n-2)……+F(n-(m-1)) +F(n-m)
令n=n-1
F(n-1)=F(n-2)……+F(n-m)+F(n-1-m)
解释:因为n>m,即n-m>0,又因为他们是整数,所以n-m>=1,所以当n=n-1时,它能走的最大阶数仍为m,所以有F(n-1-m)。
两式相减即可得到
F(n)=2*F(n-1)-F(n-1-m)
这简直就是在做数学。
当m>=n时,即能跳的最大级数大于等于总台阶数,这时我们采用解决问题二的公式即可。
//n为总阶数,m为能跳的最大阶数
int RJump3(int n, int m)
{
//
if (n == 1||n == 0)
return 1;
//当m小于n时可采用上述递推公式
if (m < n)
{
return 2 * RJump3(n - 1, m) - RJump3(n - 1 - m, m);
}
//当m大于等于n时可采用第二题的公式
else
{
//注意此时传参,青蛙能跳的阶数最大只能为总的阶数,即n
return 2 * RJump3(n - 1, n);
}
}
void test()
{
printf("第一种的递归:%d\n", RJump1(4));
printf("第一种的非递归:%d\n", Jump1(4));
printf("第二种的递归:%d\n", RJump2(4));
printf("第三种的递归,n>m:%d\n", RJump3(4, 3));
printf("第三种的递归, n, RJump3(3, 4));
}
int main()
{
test();
return 0;
}