关于微软面试题:"四人过桥"问题的思考——“n人过桥”问题的演进(Java实现)

前段时间,在CSDN上无意中看到有人发表了一道微软面试题——“4人过桥”问题的算法,只是一直忙于手头上的工作,没有能够在短期实现。当时就是想,如果把这个问题用计算机语言实现的话,有些多此一举,即使我们用穷举法,也只不过是18种情况。所以就没有深入思考,一直没有用Java来实现具体代码。

问题是这样的:

问题:四人夜过桥,步行时间分别为 1、2、5、10 分钟,四人只有一台手电筒,一趟最多两人过桥,一趟过桥须持手电筒,时间以最慢者计,问 17 分钟内可否过桥,如何过桥?

(仅仅针对4人过桥,可以参考http://blog.csdn.net/hikaliv/archive/2009/08/24/4479956.aspx

今天无意中想起递归算法,突生灵感,如果用Java语言实现“n人过桥”问题,那就有意思了。

递归的出口是:“2人过桥”情况。2人过桥,不需要有人返回,所以非常简单,总时间就是单人所需时间中的最大值。

如果是“n人过桥”(n>=3),那完全可以递归了。

假设是从桥头A至桥头B,桥头A的人群为一个集合,桥头B的人群为另一个集合。

那么首先可以从A中任意选择2个人从A到B;则A集合中减少2个人,B集合中增加2个人;

然后需要一个人从B返回A,这个可以分析出如果想要比较少的时间,一定是从B中选一个单独需时最短的;此时B中减少一个人,A集合中增加一个人;

之后情况变成了“n-1人过桥”问题。

递归思想就开始起作用了。

但是需要注意一点,我在这里的思想是每次返回的人都是从B集合中选出需时最少的;如果想找出需时最多的,就从B中选出一个需时最大的;如果想找到所有情况,那就需要遍历B集合,那就比较复杂了,我没有考虑。

如下是我的代码,由于时间仓促,没有规范化处理。。。比如passMethod中的参数列表中第三个参数其实是可以去掉的,因为它就是桥头A端得人数,可以从第一个参数中获得。

由于时间有限,我在这里就不改动了。

读者可以自己个界面,允许人/机交互,如果深入思考,还是很有意思的。

在测试代码中(main方法中),我们可以自己随意设置source参数,这里我还是沿用了“4人过桥”原参数。

控制台输出结果为:

A->B:1 AND 2
B->A:1
A->B:5 AND 10
B->A:2
A->B:1 AND 2
*****Total Time: 17****
A->B:5 AND 1
B->A:1
A->B:10 AND 1
*****Total Time: 19****
A->B:10 AND 1
B->A:1
A->B:5 AND 1
*****Total Time: 19****
A->B:1 AND 5
B->A:1
A->B:2 AND 10
B->A:2
A->B:1 AND 2
*****Total Time: 20****
A->B:2 AND 1
B->A:1
A->B:10 AND 1
*****Total Time: 19****
A->B:10 AND 1
B->A:1
A->B:2 AND 1
*****Total Time: 19****
A->B:1 AND 10
B->A:1
A->B:2 AND 5
B->A:2
A->B:1 AND 2
*****Total Time: 20****
A->B:2 AND 1
B->A:1
A->B:5 AND 1
*****Total Time: 19****
A->B:5 AND 1
B->A:1
A->B:2 AND 1
*****Total Time: 19****
A->B:2 AND 5
B->A:2
A->B:1 AND 10
B->A:1
A->B:2 AND 1
*****Total Time: 20****
A->B:1 AND 2
B->A:1
A->B:10 AND 1
*****Total Time: 20****
A->B:10 AND 2
B->A:2
A->B:1 AND 2
*****Total Time: 21****
A->B:2 AND 10
B->A:2
A->B:1 AND 5
B->A:1
A->B:2 AND 1
*****Total Time: 20****
A->B:1 AND 2
B->A:1
A->B:5 AND 1
*****Total Time: 20****
A->B:5 AND 2
B->A:2
A->B:1 AND 2
*****Total Time: 21****
A->B:5 AND 10
B->A:5
A->B:1 AND 2
B->A:1
A->B:5 AND 1
*****Total Time: 23****
A->B:1 AND 5
B->A:1
A->B:2 AND 1
*****Total Time: 23****
A->B:2 AND 5
B->A:2
A->B:1 AND 2
*****Total Time: 24****

猜你喜欢

转载自blog.csdn.net/fengyud/article/details/4647139