一类void 递归函数的非递归实现
来源:互联网 发布:雷阿伦雄鹿时期数据 编辑:程序博客网 时间:2024/06/06 09:05
对于如下类型的void型递归函数:(主要特征是递归调用的地方上下文无关)
void Fun(type a1,type a2......) { //0号程序段-起 //0号程序段-止 Fun(b1,b2,.....); //1号程序段-起 -----注意0,1,...号程序段之间并无上下文关联(没有共享局部变量) //1号程序段-止 Fun(c1,c2,.....); //2号程序段-起 //2号程序段-止......... }
将函数(程序段)一次调用视为一个节点,就可以将递归函数整个一次的执行看做是树的遍历:
如函数:
void Test(int n){ if(n>1){ Test(n-1); printf("Test%d\n",n); Test(n-2); } }
Test(n)函数
O (root节点) ----------Test函数,即if(n>1){ A }
|
O ----------A
/ | \
O O O
Test(n-1) Printf... Test(n-2)
继续向下... 执行 继续向下.....
将图中的节点编号,Test函数节点为-1;A程序段编号为0;printf程序段编号为1;
struct Param{ int number; int n;}void Test(int n){ stack<Param> ss; Param p; p.number = -1;p.n = n; ss.put(p); while(!ss.empty) { Param pa = ss.top(); ss.pop();//原递归函数为void型,不需要回溯 switch(ss.number){ case -1://-1号节点Test函数节点 if(ss.n>1){ pa.number = 0; pa.n = n; ss.push(pa); } break; case 0://0号节点A程序段节点,即:Test(n-1);printf("Test%d\n",n);Test(n-2); //Test(n-2)节点压入栈中 pa.number = -1; pa.n = n-2; ss.push(pa); //printf程序段压入栈中 pa.number = 1; //pa.n = 0;//这个参数没有意义 ss.push(pa); //Test(n-1)节点压入栈中 pa.number = -1; pa.n = n-1; ss.push(pa); break; case 1://printf函数段,即:printf("Test%d\n",n); printf("Test%d\n",n); break; } }}
稍作简化,因为-1号节点后继节点只有一个节点,直接将两个节点合并(使用合并可能不是特别贴切)了:
Test(n)函数
O (root节点) ----------Test函数
/ | \
O O O
Test(n-1) Printf... Test(n-2)
继续向下... 执行 继续向下.....
将Test节点编号为-1,Printf节点编号为0;
void Test(int n){ stack<Param> ss; Param p; p.number = -1;p.n = n; ss.put(p); while(!ss.empty) { Param pa = ss.top(); ss.pop();//原递归函数为void型,不需要回溯 switch(ss.number){ case -1://-1号节点Test函数节点 if(ss.n>1){ //Test(n-2)节点压入栈中 pa.number = -1; pa.n = n-2; ss.push(pa); //printf程序段压入栈中 pa.number = 1; //pa.n = 0;//这个参数没有意义 ss.push(pa); //Test(n-1)节点压入栈中 pa.number = -1; pa.n = n-1; ss.push(pa); } break; case 0://printf函数段,即:printf("Test%d\n",n); printf("Test%d\n",n); break; } }}
说明:
1、注意压栈的顺序,反向压栈
2、本方法只是用于递归调用的地方不需要保存现场(Test函数递归调用前后没有关联)
如果Test函数是这样的:
void Test(int n){ if(n>1){ int t = 1; Test(n-1); t = 0; Test(n-2); }}
由于t变量的存在,因此需要额外的空间保存t的值(天知道t需不需要使用到),不属于本文讨论的此类递归函数
3、递归函数不能有返回值(之后可能会讨论这样的递归函数)
- 一类void 递归函数的非递归实现
- 某些函数的递归与非递归实现的比较
- ackerman函数的非递归实现(递归函数非递归化)
- Fibonacci函数的递归和非递归实现
- ackman函数的非递归实现
- Ackerman函数的非递归实现.
- 组合函数递归和非递归实现
- 递归和非递归实现规律函数
- 递归函数的非递归化
- 递归方法的非递归实现
- 递归非递归实现树的遍历
- 递归方法的非递归实现
- Mergersort的递归及非递归实现
- 树的递归与非递归实现
- 阶乘的递归和非递归实现
- Fibonacci的递归及非递归实现
- 排列的非递归实现
- 【计组】MIPS实现的递归函数的非递归实现
- c/c++中const的常用方法
- 基本数据类型对象包装类
- DirectX的Vertex Buffer顶点缓冲的理解和应用 Shader必知必会
- 汇编语言中PTR的含义及作用
- Windows server 2008 II7下架设ASP网站标准流程
- 一类void 递归函数的非递归实现
- php发布webservice
- 进程与线程的一个简单解释
- iOS编程——1种下拉效果(渐变)
- poj_2503_Babelfish(字典树&&map)
- POJ P1611 The Suspects 基础并查集
- TCP和UDP之间的区别和联系
- linux系统命令之文件命令
- 进程和线程的区别