思路
模拟递归程序执行过程,借助一个堆栈,把递归转成非递归算法。
转化过程
1. 递归算法
1 void hanoi(int n, char from, char pass, char to) {
2 if (n == 0)
3 return;
4
5 hanoi(n - 1, from, to, pass);
6 move(n, from, to);
7 hanoi(n - 1, pass, from, to);
8 }
2. 处理首递归
本函数第2行是结束条件,第5行开始进入首递归。执行第5行函数调用之前,需要保留调用现场,本例中是4个参数入栈,使用新的参数调用hanoi函数。而继续跟踪被调用的函数,可以看出需要一直进行入栈操作,直到参数n == 0为止。 对此行为,我们可以用一个循环来模拟:
伪代码:
int i = n;
while (i != 0) {
push(i);
i --;
}
现在可以断言 i ==0 ,满足函数返回的条件。当函数返回后,需要通过pop操作来恢复现场,然后继续执行后面的语句。为了简化问题,我们假设后面只有move()一条语句,执行完毕该语句后就继续向上一层回溯,直至最顶层, 这样又可以用一个循环来模拟:
伪代码:
int i = n;
while (i != 0) {
push(i);
i --;
}
while (栈不空) {
int m = pop();
move(m, ...);
尾递归...
}
3. 处理尾递归
一般而言,尾递归可以直接改成上一条语句的循环。但在本例中,尾递归被嵌在另一个循环中,此时需要模拟它的行为:进入尾递归hanoi()函数后,需要执行其函数体,而函数体又是上述代码中的第一句话,可以如下表示:
伪代码:
a_new_beginning:
int i = n;
while (i != 0) {
push(i);
i --;
}
while (栈不空) {
int m = pop();
move(m, ...);
产生新的参数,跳出循环,跳转到a_new_beginning语句处;
}
相比于第一个while全部执行,第二个while实际只被执行一次就跳出来了,这种结构,很显然可以等价变换为外加一个大循环。那么在外部的大循环的作用下,第二个while循环可以“降维”, 去掉一层循环, 并根据pop()的现场来产生新的一批参数,给下一次循环使用。注意,两层循环合并后,循环终止条件也需要进行合并。
伪代码:
while (!(n == 0 && 栈空)) // 内外两层终止条件进行合并
int i = n;
while (i != 0) {
push(i);
i --;
}
int m = pop();
move(m, ...);
//产生新的参数
n = m;
n --; // 对应hanoi(n - 1, pass, ...)
}
最后进行完善,加上其他参数的变换。
源代码
#include <iostream>
#include <stack>
using namespace std;
void move(int n, char from, char to) {
cout << "from " << from << " move " << n << " to " << to << endl;
}
struct param {
int n;
char from;
char pass;
char to;
};
void hanoi(int n, char from, char pass, char to) {
stack<param> s;
param par_outer = {n, from, pass, to};
while (!(par_outer.n == 0 && s.empty())) {
param par_inner = par_outer;
while (par_inner.n > 0) {
s.push(par_inner);
par_inner.n --;
swap(par_inner.pass, par_inner.to);
}
par_outer = s.top();
s.pop();
move(par_outer.n, par_outer.from, par_outer.to);
par_outer.n --;
swap(par_outer.from, par_outer.pass);
}
}
int main() {
int N = 5;
hanoi(N, 'A', 'B', 'C');
return 0;
}