一类void 递归函数的非递归实现

        对于如下类型的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、递归函数不能有返回值(之后可能会讨论这样的递归函数)




  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
非递归后序遍历二叉树的实现需要借助实现。具体步骤如下: 1. 定义一个,初始化为空。 2. 将根节点入。 3. 循环执行以下步骤,直到为空: 1. 取出顶节点,将其值保存到结果数组中。 2. 如果该节点有左子节点,则将左子节点入。 3. 如果该节点有右子节点,则将右子节点入。 4. 将结果数组反转,得到后序遍历序列。 下面是使用类c语言实现的代码: ``` typedef struct TreeNode { int val; struct TreeNode* left; struct TreeNode* right; } TreeNode; void postorderTraversal(TreeNode* root, int* returnSize, int* ret) { if (root == NULL) { *returnSize = 0; return; } int len = 0; // 结果数组的长度 int* res = (int*)malloc(sizeof(int) * 100); // 初始化结果数组 TreeNode** stack = (TreeNode**)malloc(sizeof(TreeNode*) * 100); // 定义 int top = -1; // 顶指针 stack[++top] = root; // 根节点入 while (top >= 0) { TreeNode* node = stack[top--]; // 取出顶节点 res[len++] = node->val; // 将节点的值保存到结果数组中 if (node->left != NULL) { stack[++top] = node->left; // 如果该节点有左子节点,则将左子节点入 } if (node->right != NULL) { stack[++top] = node->right; // 如果该节点有右子节点,则将右子节点入 } } // 反转结果数组 int i = 0, j = len - 1; while (i < j) { int temp = res[i]; res[i++] = res[j]; res[j--] = temp; } // 将结果数组返回 *returnSize = len; for (i = 0; i < len; i++) { ret[i] = res[i]; } } ``` 其中,`root` 表示二叉树的根节点,`returnSize` 表示结果数组的长度,`ret` 表示结果数组。在函数中,先判断根节点是否为空,如果是,则直接返回空数组;否则,初始化结果数组和,并将根节点入。接着,循环取出顶节点,并将其值保存到结果数组中。如果该节点有左子节点,则将左子节点入;如果该节点有右子节点,则将右子节点入。循环结束后,反转结果数组,并将其返回。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值