二叉树的遍历(非递归)—— 基于栈的递归消除

目录

一、DLR先序遍历

1.1 算法实现

1.2 算法思想

二、LDR中序遍历 

2.1 算法实现

2.2 算法思想 

三、LRD后序遍历

3.1 算法实现

3.2 算法思想


一、DLR先序遍历

1.1 算法实现

void InOrder(BiTree root, CALLBACK Visit) {
    Stack s, *S = &s;
    BiTree p;

    InitStack(S);

    p = root;

    while (p != NULL || !IsEmpty(S)) {
        if (p != NULL) {
            Visit(p->data);
            Push(S, &p); /* 留下面包屑 */
            p = p->LChild;
        } else {
            Pop(S, &p); /* 弹出结点,由p来指向 */
            p = p->RChild;
        }
    }

    ClearStack(S);
}

1.2 算法思想


二、LDR中序遍历 

2.1 算法实现

void InOrder(BiTree root, CALLBACK Visit) {
    Stack s, *S = &s;
    BiTree p;

    InitStack(S);

    p = root;

    while (p != NULL || !IsEmpty(S)) {
        if (p != NULL) {
            Push(S, &p); /* 留下面包屑 */
            p = p->LChild;
        } else {
            Pop(S, &p); /* 弹出结点,由p来指向 */
            Visit(p->data);
            p = p->RChild;
        }
    }

    ClearStack(S);
}

2.2 算法思想 

 


三、LRD后序遍历

与先序、中序遍历算法的区别:

1、增加结点指针q对已访问的结点进行标记,避免重复访问。

2、使用GetTop函数获取栈顶结点,避免一次性弹出后无法返回到子树根结点。

3.1 算法实现

void PostOrder(BiTree root, CALLBACK Visit) {
    BiTNode *p, *q;
    Stack s, *S = &s;

    q = NULL;
    p = root;

    InitStack(S);

    while (p != NULL || !IsEmpty(S)) {
        if (p != NULL) {
            Push(S, &p); /* 留下面包屑 */
            p = p->LChild;
        } else {
            GetTop(S, &p); /* 获取栈顶结点 */
            if (p->RChild == NULL || p->RChild == q) {
                Visit(p->data);
                q = p; /* 对已访问结点进行标记 */
                Pop(S, &p); /* 同时弹出已访问结点 */
                p == NULL;
            } else {
                p = p->RChild;
            }
        }
    }

    ClearStack(S);
}

3.2 算法思想 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
归先序遍叉树的思路是使用来模拟归过程。具体的代码如下所示: ``` public static void preOrderTraverse(TreeNode root) { if (root == null) return; Stack<TreeNode> stack = new Stack<>(); stack.push(root); while (!stack.isEmpty()) { TreeNode cur = stack.pop(); System.out.println(cur.val); if (cur.right != null) stack.push(cur.right); if (cur.left != null) stack.push(cur.left); } } ``` 首先,我们创建一个并将根节点入。然后,我们进入一个循环,直到为空。在循环中,我们弹出顶的节点并打印其值。接着,我们将右子节点入,然后将左子节点入。因为是后进先出的,所以弹出顶的节点时,会先打印它的值,然后先压入右子节点,再压入左子节点。这样就实现了先序遍的顺序。通过这个归算法,我们可以按先序遍的顺序访问二叉树的所有节点。 举个例子来演示算法的执行过程。假设我们有如下二叉树: ``` 1 / \ 2 3 / \ \ 4 5 6 ``` 我们按照归先序遍的算法来遍这棵树。首先,我们将根节点1入。然后,进入循环,弹出顶的节点1并打印它的值。接着,我们将右子节点3入,再将左子节点2入。此时,中的节点顺序为:3, 2。接下来,我们再次进入循环,弹出顶的节点2并打印它的值。因为节点2没有子节点,所以继续弹出顶的节点3并打印它的值。此时,中只剩下根节点1了。我们再次进入循环,弹出顶的节点1并打印它的值。最后,为空,遍结束。 所以,按照归先序遍的算法,上述二叉树的遍结果为:1, 2, 4, 5, 3, 6。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [二叉树归遍算法](https://blog.csdn.net/weixin_47396585/article/details/120824248)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* *3* [归先序遍叉树总结(3种方法)](https://blog.csdn.net/qq_41512783/article/details/110524693)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值