汉罗塔c++递归_栈与递归的区别

汉罗塔c++递归_栈与递归的区别汉罗塔问题是一个非常经典的算法,我们首先来研究一下修改的汉罗塔(简化步骤),在后面我们将来讲述经典的汉罗塔问题。题目:修改后的汉罗塔的规则:现在限制不能从最左侧的塔直接移动到最右侧,必需要经过中间;同时从最右侧移动到最左测试,同样必需经过中间;要求移动N层塔时,打印最优移动1、用递归函数实现(从最左移动到最右)分析:-当只有一层塔时,我们先需要将其从左移到中间,再从中间移动到右边,共分为

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE稳定放心使用

汉罗塔问题是一个非常经典的算法,我们首先来研究一下修改的汉罗塔(简化步骤),在后面我们将来讲述经典的汉罗塔问题。

题目:
修改后的汉罗塔的规则:现在限制不能从最左侧的塔直接移动到最右侧,必需要经过中间;同时从最右侧移动到最左测试,同样必需经过中间;要求移动N层塔时,打印最优移动

1、用递归函数实现(从最左移动到最右)

分析:
– 当只有一层塔时,我们先需要将其从左移到中间,再从中间移动到右边,共分为两步;如果它就在中间,那么我们只需要将它移动到左或则右,一步就行;
– 当我们有N 层塔时,我们需要先将1~N-1层塔移动到右边,然后移动第N层塔到中间,再将1~N-1层塔移动到最左边,将N层塔由中间移动右边;这样,第N层塔就移好了
– 接下来重复上述步骤,将1~N-2层塔移到最右边,将第N-1层塔移到最中间……(利用递归函数实现)

代码实现:

void  _HanoiProblem1(int num,string from,string to,int& count)//num代表塔的层数,from代表开始移动的位置(left,mid,right),to 代表要移动到的位置
{
    if(num == 1)//我们只有一层塔
    {
        if(from.compare("mid")==0 || to.compare("mid")==0)//此时只需直接移动
        {
            count++;
            cout<<"move "<<num<<" from "<<from.c_str()<<" to "<<to.c_str()<<endl;
        }
        else
        {
            cout<<"move "<<num<<" from "<<from.c_str()<<" to "<<"mid"<<endl;
            count++;
            cout<<"move "<<num<<" from "<<"mid"<<" to "<<to.c_str()<<endl;
            count++;
        }
    }
    else//我们有多层塔
    {
        if(from.compare("mid")==0 || to.compare("mid")==0)
        {
            string another;
            if(from.compare("left")==0)
                another = "right";
            else
                another = "left";

            _HanoiProblem1(num-1,from,another,count);//例如从左移到中间,先将1~N-1层塔移动到右边,
            cout<<"move "<<num<<" from "<<from.c_str()<<" to "<<to.c_str()<<endl;//再将第N层塔移到中间
            count++;
            _HanoiProblem1(num-1,another,to,count);//再将1~N-1层塔移到中间

        }
        else//从左移到右或从右移到左
        {
            _HanoiProblem1(num-1,from,to,count);//例如从左移到右,先将1~N-1层塔从左移动到右边,
            cout<<"move "<<num<<" from "<<from.c_str()<<" to "<<"mid"<<endl;//再将第N层塔移到中间
            count++;
            _HanoiProblem1(num-1,to,from,count);//再将1~N-1层塔从右移动到左边
            cout<<"move "<<num<<" from "<<"mid"<<" to "<<to.c_str()<<endl;//再将第N层塔从中间移到右边
            count++;
            _HanoiProblem1(num-1,from,to,count);//再将1~N-1层塔从左移动到右边
        }
    }
}
void HanoiProblem1(int num,string from,string to)
{
    int count = 0;
    _HanoiProblem1(num,from,to,count);
    cout<<"count is: "<<count<<endl;
}

测试代码:

void funtest()
{
    HanoiProblem1(2,"left","right");
}

int main()
{
    funtest();
    getchar();
    return 0;
}

结果图
这里写图片描述

2.用栈模拟实现

分析:
我们上面用递归实现,我们已经知道了基本的走法,接下来我们用栈来模拟汉罗塔问题,将塔的移动转换为入栈和出栈的操作,但是,由题我们知道了参数入栈和出栈的两个基本规则

  • 小压大问题,即只有当要入栈的参数小于栈顶元素,这时我们才能入栈
  • 动作不想临,题目要求我们实现最优移动,所以我们从左移动到中间,下一步将它从中间右移动到左边,是没有意义的

满足了以上两条规则,我们现在看移动的过程,一个塔a,只有四中可能的动作,从左到中,从中到右,从右到中,从中到左,但是要满足以上两种规则我们发现它只有一种动作可以走;例如:a在最左边,第一次,它只能从左到中间,第二次,由规则知,从左到中间不行,从中间到左没有意义,那么只剩下从中间到右和从右到中间,我们只需判断就能得到结果。

代码实现:

enum action
{
    No,
    LToM,
    MToL,
    RToM,
    MToR,
};

int fStackToStack(action& record,action preAction,action nowAction,stack<int>& fstack,stack<int>& tstack,string from,string to)
{
    if(record != preAction && fstack.top() < tstack.top())//满足两条准则,动作不可逆和小压大原则
    {
        int data = fstack.top();
        fstack.pop();
        tstack.push(data);
        cout<<"move "<<data<<" from "<<from.c_str()<<" to "<<to.c_str()<<endl;
        record = nowAction;
        return 1;
    }
    return 0;
}

int HanoiProblem2(int num,string left,string mid,string right)
{
    stack<int> ls;
    stack<int> ms;
    stack<int> rs;

    ls.push(INT_MAX);
    ms.push(INT_MAX);
    rs.push(INT_MAX);

    for(int idx=num; idx>0; --idx)
        ls.push(idx);

    action record = No;//记录上一步的动作
    int count = 0;

    while(rs.size() != num+1)
    {
        count += fStackToStack(record,MToL,LToM,ls,ms,left,mid);
        count += fStackToStack(record,LToM,MToL,ms,ls,mid,left);
        count += fStackToStack(record,MToR,RToM,rs,ms,right,mid);
        count += fStackToStack(record,RToM,MToR,ms,rs,mid,right);
    }
    return count;
}

测试代码:

void funtest()
{
    int ret = HanoiProblem2(2,"left","mid","right");
    cout<<ret<<endl;
}

int main()
{
    funtest();
    getchar();
    return 0;
}

结果图:
这里写图片描述

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/183281.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • linux netstat -an命令,linux 命令之netstat[通俗易懂]

    linux netstat -an命令,linux 命令之netstat[通俗易懂]在linux中netstat命令的作用是查看TCP/IP网络当前所开放端口,所对应的本地和外地端口信息。netstat命令的格式netstat[-a][-e][-n][-o][-pProtocol][-r][-s][Interval]各参数选项的含义a显示所有socket,包括正在监听的。-c每隔1秒就重新显示一遍,直到用户中断它。-i显示所有网络接口的信息,格式“netstat-i”…

    2022年8月30日
    1
  • intellijidea激活码【最新永久激活】

    (intellijidea激活码)2021最新分享一个能用的的激活码出来,希望能帮到需要激活的朋友。目前这个是能用的,但是用的人多了之后也会失效,会不定时更新的,大家持续关注此网站~IntelliJ2021最新激活注册码,破解教程可免费永久激活,亲测有效,下面是详细链接哦~https://javaforall.net/100143.html…

    2022年3月28日
    173
  • visio2010密钥

    visio2010密钥GR24B-GC2XY-KRXRG-2TRJJ-4X7DCVWQ6G-37WBG-J7DJP-CY66Y-V278X2T8H8-JPW3D-CJGRK-3HTVF-VWD83HMCVF-BX8YB-JK46P-DP3KJ-9DRB222WT8-GGT7M-7MVKR-HF7Y4-MCWWDVX6BF-BHVDV-MHQ4R-KH9QD-6TQKVJ4MVP-7F4X4-V8W2C-…

    2022年5月29日
    602
  • nginx实现负载均衡几种方式_nginx如何负载均衡

    nginx实现负载均衡几种方式_nginx如何负载均衡Nginx负载均衡配置实例详解(转)负载均衡是我们大流量网站要做的一个东西,下面我来给大家介绍在Nginx服务器上进行负载均衡配置方法,希望对有需要的同学有所帮助哦。负载均衡先来简单了解一下什么是负载均衡,单从字面上的意思来理解就可以解释N台服务器平均分担负载,不会因为某台服务器负载高宕机而某台服务器闲置的情况。那么负载均衡的前提就是要有多台服务器才能实现,也就是两台以上即可。测试环境由于没有服务器,所以本次测试直接host指定域名,然后在VMware里安装了三台CentOS。测试域名

    2025年6月3日
    0
  • C++学习——虚函数与纯虚函数

    C++学习——虚函数与纯虚函数引言:若要访问派生类中相同名字的函数,必须将基类中的同名函数定义为 虚函数,这样,将不同的派生类对象的地址赋给基类的指针变量后, 就可以动态地根据这种赋值语句调用不同类中的函数。一、虚函数的定义和使用可以在程序运行时通过调用相同的函数名而实现不同功能的 函数称为虚函数。定义格式为:virtual FuncName();一旦把基类的成员函数定义为虚函数,由基类所派生出来的所 有派生类中,…

    2022年8月18日
    8
  • js统计英文单词数量

    js统计英文单词数量js单词数量

    2025年5月26日
    0

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号