面试算法:使用堆栈解决汉诺塔问题

更详细的讲解和代码调试演示过程,请参看视频
如何进入google,算法面试技能全面提升指南

在算法研究中,汉诺塔是非常经典的一道题。主要是它的求解过程,所展现的思维方式极具代表性,它的解法是,你要解决一个大问题,首先把大问题化解成几个容易解决的小问题,把小问题的解决方案进行简单的操作组合就能得到大问题的解,解决小问题的时候,就是把小问题分解成更小的问题,就这么一直分解,直到问题被分解到轻而易举就能处理的程度。

汉诺塔问题就是这种先分解后综合的典型代表。

面试算法:使用堆栈解决汉诺塔问题_第1张图片
这里写图片描述

题目是这样的,有三根杆,其中一根杆子上有n个铁饼,铁饼由小到大依次从上到下排列,如上图。要求你把杆子1上的铁饼挪到杆子2,杆子3可以作为转移铁饼的中转站,当你转移铁饼时,必须保证小铁饼只能放到大铁饼的上头,请你给出转移铁饼的步骤。

要想把n块铁饼挪到杆子2,我们可以把问题分解成小一点的问题,例如我们可以先把n-1 个铁饼挪到杆子3,然后把最后一个铁饼挪到杆子2,再把n-1个铁饼从杆子3挪到杆子2. 于是原来的问题规模是n,我们把它分解成了问题规模为n-1的子问题。

于是问题就转换为如何把n-1个铁饼转移到另一根杆子, 转移n-1个铁饼又可以分解成转移n-2个铁饼,就这么不断的分解下去,最后问题归结为转移1个铁饼,转移1个铁饼,那肯定就是不费吹灰之力就可以完成的。

按照这个设想,我们实现的代码如下:

import java.util.Stack;


public class HanoiMove {
    
    private int from = 0;
    private int to = 0;
    
    private Stack hanoiMove = new Stack();
    public  HanoiMove(int n, int from, int to) throws Exception {
        
        if (n <= 0 || from == to || from < 0 || from > n || to < 0 || to > n) {
            throw new Exception("Invalid parameters");
        }
        
        this.from = from;
        this.to = to;
        
        moveHanoiStack(this.from, this.to, 1, n);
    }
    
    public void printMoveSteps() {
        if (hanoiMove.size() == 1) {
            System.out.println(hanoiMove.pop());
            return;
        }
        
        String s = hanoiMove.pop();
        printMoveSteps();
        System.out.println(s);
    }
    
    
    
    private void moveHanoiStack(int from, int to, int top, int end) {
        String s = "Move ring  " + end + " from stack " + from + " to " + to;
        if (end - top == 0) {
            hanoiMove.push(s);
            return;
        }
        
        int other = from;
        for (int i = 1; i <= 3; i++) {
            if (i != from && i != to) {
                other = i;
                break;
            }
        }
        
        moveHanoiStack(from, other, top, end - 1);
        hanoiMove.push(s);    
        moveHanoiStack(other, to, top, end - 1);
    }
}

初始化函数HanoiMove需要输入三个参数,第一个参数是要移动的铁饼数量,第二个参数是表示铁饼原来所在的杆子编号,第三个参数表示要移动的目标杆子编号。例如要把杆子1的3个铁饼转移到杆子2,那么初始化函数可以这么写:

HanoiMove(3, 1, 2);

moveHanoiStack用来实现前面我们所讲的转移步骤,它的输入参数这么解读,from表示铁饼所在的杆子编号,to表示铁饼要转移的杆子编号,top表示最上头的铁饼编号,end 表示最下方的铁饼编号,例如,把杆子1上的三块铁饼搬到杆子2,那么对应的调用就是:

moveHanoiStack(1, 2, 1, 3)

因为杆子1有三块铁饼,最上方也就是最小的铁饼,用编号1表示,最下方,也就是最大的铁饼,用编号3表示。

如果top 和 end 相等,那意味着我们只需要搬运一块铁饼,那么这种简单情况,我们直接把铁饼搬过去就可以了。如果搬运的是n块,那么我们先找到中转杆子,这就是for循环要做的事情,for的目的是遍历所有杆子,找到一根杆子,它的编号与from和to不同即可,这根杆子我们用other表示。

于是我们就先把n-1 个铁饼从杆子from 转移到杆子 other, 然后把最底下的铁饼从杆子from 先放到杆子to, 最后再把挪到杆子other上的n-1个铁饼挪到杆子to上,这样我们就实现了把铁饼从杆子from转移到杆子to的目的。

我们看看主入口函数的实现:

public class StackAndQuque {
    public static void main(String[] args) {
        
        try {
            HanoiMove hanoi = new HanoiMove(3, 1, 2);
            hanoi.printMoveSteps();
        } catch (Exception e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }
}

代码模拟的是把杆子1上的三个铁饼挪到杆子2,初始情况如下:
1
2
3
== == ==

1,2,3代表三个铁饼,下面的横线代表三个杆子。上面代码运行后,每一次铁饼转移的方式都已字符串的形式记录在一个栈中,栈最底部的字符串记录第一步转移,倒数第二个字符串记录第二部铁饼的转移,以此类推。

代码的运行结果如下:
Move ring 1 from stack 1 to 2
Move ring 2 from stack 1 to 3
Move ring 1 from stack 2 to 3
Move ring 3 from stack 1 to 2
Move ring 1 from stack 3 to 1
Move ring 2 from stack 3 to 2
Move ring 1 from stack 1 to 2

我们根据上面的步骤把铁饼挪以此,走第一步后情况如下:
2
3 1
== == ==

走第二步,把2号铁饼从杆子1挪到杆子3:
3 1 2
== == ==

走第三部,把1号铁饼从杆子2 挪到杆子3:
3 1
2
== == ==

走第四步,把3号铁饼从杆子1挪到杆子2:
1

   3      2

== == ==

走第五步,把1号铁饼从杆子3移到杆子1:
1 3 2
== == ==

走第六步,把2号铁饼从杆子3移到杆子2:
1 2
3
== == ==

走第七步,把1号铁饼从杆子1挪到杆子2:
1
2
3
== == ==

由此,我们正确的把杆子1上的三个铁饼挪到了杆子2,也就是我们的算法是正确的。

汉诺塔问题是NP完全问题,也就是说,它的算法时间复杂度是指数级的,分析如下,假设我们要挪动n个铁饼,把时间标记为T(n), 我们先挪动n-1个铁饼,所需时间就是T(n-1), 然后再挪动一个铁饼,时间为O(1), 然后再挪动n-1个铁饼,时间为T(n-1), 于是我们有:
T(n) = 2*T(n-1) + O(1).
这个公式把T(n)解出来结果为: T(n) = 2^n;

这就意味着,每增加一个铁饼,所需的挪动步骤几乎是原来的两倍。如果我们把入口函数铁饼数3改成4,所得结果如下:
Move ring 1 from stack 1 to 3
Move ring 2 from stack 1 to 2
Move ring 1 from stack 3 to 2
Move ring 3 from stack 1 to 3
Move ring 1 from stack 2 to 1
Move ring 2 from stack 2 to 3
Move ring 1 from stack 1 to 3
Move ring 4 from stack 1 to 2
Move ring 1 from stack 3 to 2
Move ring 2 from stack 3 to 1
Move ring 1 from stack 2 to 1
Move ring 3 from stack 3 to 2
Move ring 1 from stack 1 to 3
Move ring 2 from stack 1 to 2
Move ring 1 from stack 3 to 2

挪动4个铁饼需要15步,恰巧是挪动3个铁饼所需要步骤数的2倍。

本算法中,使用堆栈在一定程度上是多余的,如果去掉堆栈,您知道怎么把这些转移步骤打印出来吗?

更详细的讲解和代码演示,请参看视频。

更多技术信息,包括操作系统,编译器,面试算法,机器学习,人工智能,请关照我的公众号:


面试算法:使用堆栈解决汉诺塔问题_第2张图片
这里写图片描述

你可能感兴趣的:(面试算法:使用堆栈解决汉诺塔问题)