美文网首页
用栈来求解汉诺塔问题

用栈来求解汉诺塔问题

作者: 永志 | 来源:发表于2016-07-30 10:15 被阅读0次

题目

在汉诺塔规则的基础上,限制不能从最左的塔移动到最右的塔上,必须经过中间的塔,移动的跨度只能是一个塔。当塔有N层的时候,打印最优移动过程和最优移动步数。

要求

  • 方法一:使用递归的方法进行移动
  • 方法二:使用栈进行移动

解答思路

方法一:

无论多少层,都看作有两层,最大的一层(命名为X)、(N-1)层合并起来的作为一层(命名为Y),目标是将X移动到最右侧,然后再把Y移动到最右侧。

汉诺塔
递归的移动方式:
  • Y从A塔移动到B塔
  • Y从B塔移动到C塔
  • X从A塔移动到B塔
  • Y从C塔移动到B塔
  • Y从B塔移动到A塔
  • X从B塔移动到C塔
  • 将Y看做X,继续递归移动
实现代码:
import java.util.Stack;

/**
 * 每次移动只能移动一个柱子,不能跨柱子移动
 * @author zhanyongzhi
 */
public class HannoiOneStep {
    public void startMove(int count){
        move(count, "A", "B", "C");
    }

    private void move(int item, String from, String buffer, String to){
        //
        if(1 == item){
            System.out.println(String.format("move %d from %s to %s", item, from, buffer));
            System.out.println(String.format("move %d from %s to %s", item, buffer, to));
            return;
        }

        //general situation
        move(item - 1, from, buffer, to);
        System.out.println(String.format("move %d from %s to %s", item, from, buffer));
        move(item - 1, to, buffer, from);
        System.out.println(String.format("move %d from %s to %s", item, buffer, to));

        move(item - 1, from, buffer, to);
    }
}

方法二:

使用栈而不使用递归的方式进行移动,使用3个栈模拟3个塔,每一步的移动,都按照真实情况进行。
  按照规则,可能的移动动作限定为LM、ML、MR、RM四种步骤(L、M、R分布表示左中右),通过引入逆反原则和小压大原则,可以得出每次移动,只有一种可行步骤。

逆反原则

当执行了LM,如果此时下一步执行ML,叫做逆反操作,这样会使得汉诺塔还原为上一步的形状,白走多一步,这样明显不是最优的方法,所以不能够执行逆反操作,叫逆反原则。

小压大原则

当移动时,小的块总是在大块之上,叫小压大原则。

限制分析

当上一步为:LM,下一步的情况分析:

  • 执行LM,违反小压大原则
  • 执行ML,违反逆反原则
  • 执行MR还是RM,按照小压大原则,这两种情况是互斥的,只能按条件二选一

其他分析类似,省略...

实现代码

package com.github.zhanyongzhi.interview.algorithm.stacklist;

import java.util.Stack;

/**
 * 使用栈模拟汉诺塔移动,将towerA全部层移动到towerC
 * @author zhanyongzhi
 */
public class HannoiStack {
    private Stack<Integer> towerA = new Stack<>();
    private Stack<Integer> towerB = new Stack<>();
    private Stack<Integer> towerC = new Stack<>();

    private MoveType preMoveType = MoveType.LM;

    enum MoveType{
        LM("Move From Left to Middle"),
        MR("Move From Middle to Right"),
        RM("Move From Right to Middle"),
        ML("Move From Middle to Left");

        private final String name;

        MoveType(String s) {
            name = s;
        }

        public boolean equalsName(String otherName) {
            return (otherName == null) ? false : name.equals(otherName);
        }

        public String toString() {
            return name;
        }
    }

    public void init(int size){
        for(int i=size; 0 < i; i--){
            towerA.push(i);
        }
    }

    public void startMove(){
        int layerSize = towerA.size();

        while(layerSize != towerC.size()){
            moveStack(MoveType.LM, MoveType.ML, towerA, towerB);
            moveStack(MoveType.MR, MoveType.RM, towerB, towerC);
            moveStack(MoveType.RM, MoveType.MR, towerC, towerB);
            moveStack(MoveType.ML, MoveType.LM, towerB, towerA);
        }
    }

    private void moveStack(MoveType tryMove, MoveType preventMove, final Stack<Integer> towerFrom, final Stack<Integer> towerTo){
        if(preMoveType == preventMove)
            return;

        if(towerFrom.empty())
            return;

        Integer sElement = towerFrom.peek();

        if(!towerTo.empty()){
            Integer dElement = towerTo.peek();

            if(sElement > dElement)
                return;
        }

        preMoveType = tryMove;

        System.out.println(tryMove);
        towerFrom.pop();
        towerTo.push(sElement);
    }
}

在github中查看

相关文章

  • 用栈来求解汉诺塔问题

    【题目】 汉诺塔问题比较经典,这里修改一下游戏规则:现在限制不能从最左侧的塔直接移动到最右侧,也不能从最右侧直接移...

  • 用栈来求解汉诺塔问题

    题目 在汉诺塔规则的基础上,限制不能从最左的塔移动到最右的塔上,必须经过中间的塔,移动的跨度只能是一个塔。当塔有N...

  • 递归求解汉诺塔问题

    数据结构习题解析・邓俊峰 课后习题 问题有三根杆子A,B,C。A杆上有N个(N>1)穿孔圆盘,盘的尺寸由下到上依次...

  • 汉诺塔问题(递归求解)

  • 递归算法求解汉诺塔问题

    Hanoi(汉诺)塔问题,这是一个古典的数学问题。古印度有一个梵塔,塔内有3个柱子A,B,C,开始时A柱上套有64...

  • 汉诺塔-Java实现

    汉诺塔总步数:2^n -1 引入栈来表示移动wafer步骤 程序打印结果

  • 用栈取代函数递归

    文章目录 1.用栈实现汉诺塔问题: DOMOVE相当于一步移动,DOTOH相当于一次递归 2.用栈实现求斐波拉契数...

  • 用栈代替递归之汉诺塔问题

    问题描述 递归解法 运行结果 手工解法 非递归解法 运行结果 递归中的栈

  • 汉诺塔递归求解

    相关链接 汉诺塔的移动也可以看做是递归函数。我们对柱子编号为a, b, c,将所有圆盘从a移到c可以描述为:如果a...

  • 汉诺塔问题与递归

    文章也同时在个人博客 http://kimihe.com/更新 汉诺塔问题(Hanoi Tower) 汉诺塔问题的...

网友评论

      本文标题:用栈来求解汉诺塔问题

      本文链接:https://www.haomeiwen.com/subject/evpzjttx.html