递归入门

一:什么是递归

所谓递归,简单点来说,就是一个函数直接或间接调用自身的一种方法,它通常把一个大型复杂的问题层层转化为一个与原问题相似的规模较小的问题来求解。

我们可以把” 递归 “比喻成 “查字典 “,当你查一个词,发现这个词的解释中某个词仍然不懂,于是你开始查这第二个词。

可惜,第二个词里仍然有不懂的词,于是查第三个词,这样查下去,直到有一个词的解释是你完全能看懂的,那么递归走到了尽头,然后你开始后退,逐个明白之前查过的每一个词,最终,你明白了最开始那个词的意思。(摘自知乎的一个回答)

我们以阶乘作为:

int Factorial(int n){    

      if (n == 0)  return 1;  

      return

      n * Factorial(n - 1);

}

汉诺塔 - 问题起源

法国数学家爱德华·卢卡斯曾编写过一个印度的古老传说:在世界中心贝拿勒斯(在印度北部)的圣庙里,一块黄铜板上插着三根宝石针。印度教的主神梵天在创造世界的时候,在其中一根针上从下到上地穿好了由大到小的64片金片,这就是所谓的汉诺塔。不论白天黑夜,总有一个僧侣在按照下面的法则移动这些金片:一次只移动一片,不管在哪根针上,小片必须在大片上面。僧侣们预言,当所有的金片都从梵天穿好的那根针上移到另外一根针上时,世界就将在一声霹雳中消灭,而梵塔、庙宇和众生也都将同归于尽。

汉诺塔游戏
后来,这个传说就演变为汉诺塔游戏:
1.有三根杆子A,B,C。A杆上有若干碟子
2.每次移动一块碟子,小的只能叠在大的上面
3.把所有碟子从A杆全部移到C杆上

解决思路

A柱子有7块盘子时,可借助B将A柱子的盘子全部移至C柱子,由小到大从上往下摆放

游戏的起始状态,A柱子7块盘子,移至C柱子:

要实现游戏,需将A柱子红色盘子上面的6块借助C移至B柱子,盘子由小到大从上往下摆放,在将最大的红色盘子由A柱子移至C柱子,此时A柱子没有盘子,这个时候是游戏的一个临界状态,最大的红色盘子有序的摆放在C柱子(现在可以不管红色盘子了,因为最大的如果你把它放那不动,则可以无视它的,其他盘子可以在三个柱子上移来移去.它是最大嘛),接着需要将B柱子的6块盘子借助A移动到C柱子,即可完成汉诺塔游戏了。这个过程中,7块盘子的汉诺塔游戏可演变为2个6块盘子的汉诺塔游戏了,2个6块盘子的汉诺塔游戏了演变成4个5块盘子的汉诺塔游戏了……,递归的出口就是只有一块盘子的时候了,直接移动盘子,这就是经典递归思想了

游戏的临界状态,B柱子6块盘子,移动至C柱子

实现的过程:return 有时可以省去,就是不断的调用
函数
#include<stdio.h>

void doTowers(int n,char X,char Y,char Z){
    if(n == 1)
        printf("Dish 1 from %c to %c \n",X,Z);
    else{
        doTowers(n-1,X,Z,Y);   //先将n-1个盘子借助Z从X挪到Y,为了方便将最下面的盘子直接挪到目的针
        printf("Dish %d from %c to %c \n",n,X,Z);   //把最下面的盘子直接从源挪到目的
        doTowers(n-1,Y,X,Z);  //把移到Y针上的n-1个盘子再挪到最终目的针Z上
    }
}
int main(){
    int n;
    printf("Enter the count of dishes: \n");
    scanf("%d",&n);
    printf("The answer is :\n");
    doTowers(n,'X','Y','Z');         //参数分别为:盘子数、源、中介、目标
}

汉诺塔圆盘移动次数
由递归的思路可以知道,我们是先将N-1个圆盘移动从A盘移动到B盘的,再将第N个圆盘移动到C盘的,在将N-1个圆盘从B盘移动到C盘
所以从这里很容易看出N个圆盘的汉诺塔移动次数是2*(N-1)个圆盘的汉诺塔移动的次数+1
实现代码:

int  hanoi (int n){
        if(n==1){
            return 1;
        }
        return 2*hanoi(n-1)+1;
    }

hd2013 蟠桃记在这里插入图片描述

#include<iostream>
#include<stdio.h>
#include<queue>
#include<algorithm>
#include<string.h>
#include<string.h>
#include<math.h>
using  namespace std;
int peach(int n)
{
	    long res = 1;//java中后面才带L我说呢,C语言不需要 
	    
	    //后面两部我就不会 
        while(--n)
            res = 2 * (res + 1);
        return res;
}
int main()
{
	int ans=0;
	int n,a[100];
	while(cin>>n)
	{		
		cout<<peach(n)<<endl;
	}
	
	return 0;
}

参考https://blog.csdn.net/qmdweb/article/details/80537602

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值