Java 实现的二叉树的递归、非递归遍历

1.二叉树的递归遍历

//先序:根、左子树、右子树
public void preOrderRecur(Node head) {
        if (head == null) {
            return;
        }
        System.out.println(head.val);
        preOrderRecur(head.left);
        preOrderRecur(head.right);
    }
//中序:左子树、根、右子树
public void inOrderRecur(Node head) {
        if (head == null) {
            return;
        }
        inOrderRecur(head.left);
        System.out.println(head.val);
        inOrderRecur(head.right);
    }
//后序:左子树、右子树、根
public void posOrderRecur(Node head) {
        if (head == null) {
            return;
        }
        posOrderRecur(head.left);
        posOrderRecur(head.right);
        System.out.println(head.val);
    }

递归遍历的关键是,将输出语句插入到合适的位置

2.二叉树的非递归先序遍历

利用一个栈来实现先序遍历
1根节点入栈
2判断栈是否为空,如果不为空
弹出栈顶元素
打印栈顶元素
如果有右子树压栈
如果有左子树压栈
3如果栈为空,程序结束

public void preOrder(Node head) {
        if (head == null) {
            return;
        }
        Stack<Node> s = new Stack<Node>();
        s.push(head);
        while (!s.isEmpty()) {
            Node cur = s.pop();
            System.out.println(cur.val);
            if (cur.right != null) {
                s.push(cur.right);
            }
            if (cur.left != null) {
                s.push(cur.left);
            }
        }
    }

3.二叉树的非递归中序遍历

1.如果栈不为空或节点不指向空数据,则执行以下步骤
a如果节点不为空,压栈,沿着左子树走一步
b如果节点为空,则出栈,打印,沿着右子树走一步
2.如果栈为空且当前节点为空,则结束

public void inOrder(Node head) {
        if (head == null) {
            return;
        }
        Stack<Node> s = new Stack<Node>();
        while (!s.isEmpty() || head != null) {
            if (head != null) {
                s.push(head);
                head = head.left;
            } else {
                head = s.pop();
                System.out.println(head.val);
                head = head.right;
            }
        }
    }

4.二叉树的非递归后序遍历

4.1两个栈实现

1申请两个栈s1,s2,头节点入栈s1
2如果栈s1不为空,执行以下操作:弹出一个元素,入栈s2,如果该节点左孩子不空入栈s1,如果该节点右孩子不空入栈s1.
3.将栈s2中的节点一次出栈,打印。

// 两个栈实现后序遍历
    public void posOrder(Node head) {
        if (head == null) {
            return;
        }
        Stack<Node> s1 = new Stack<Node>();
        Stack<Node> s2 = new Stack<Node>();
        s1.push(head);
        while (!s1.isEmpty()) {
            Node cur = s1.pop();
            s2.push(cur);
            if (cur.left != null) {
                s1.push(cur.left);
            }
            if (cur.right != null) {
                s1.push(cur.right);
            }
        }
        while (!s2.isEmpty()) {
            Node cur = s2.pop();
            System.out.println(cur.val);
        }
    }

4.2一个栈实现后序遍历

后序遍历的难点在于:如何判断上次访问的节点是位于左子树,还是右子树,如果上一个访问的节点是左子树,那么我们则需要先跳过根节点,访问右子树,再返回根节点。

public void postOrder3(Node node){  
            if(node == null)  
                return;  
            Stack<Node> s = new Stack<Node>();  
              
            Node curNode; //当前访问的结点  
            Node lastVisitNode; //上次访问的结点  
            curNode = node;  
            lastVisitNode = null;  
              
            //把currentNode移到左子树的最下边  
            while(curNode != null){  
                s.push(curNode);  
                curNode = curNode.left;  
            }  
            while(!s.empty()){  
                curNode = s.pop();  //弹出栈顶元素  
                //一个根节点被访问的前提是:无右子树或右子树已被访问过  
                if(curNode.right!= null && curNode.right !=lastVisitNode){  
                    //根节点再次入栈  
                    s.push(curNode);  
                    //进入右子树,且可肯定右子树一定不为空  
                    curNode = curNode.right;  
                    while(curNode!= null){  
                        //再走到右子树的最左边  
                        s.push(curNode);  
                        curNode = curNode.left;  
                    }  
                }else{  
                    //访问  
                    System.out.println(curNode.val);  
                    //修改最近被访问的节点  
                    lastVisitNode = curNode;  
                }  
            } 
        }
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,015评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,262评论 1 292
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,727评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,986评论 0 205
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,363评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,610评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,871评论 2 312
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,582评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,297评论 1 242
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,551评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,053评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,385评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,035评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,079评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,841评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,648评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,550评论 2 270

推荐阅读更多精彩内容