Morris遍历(包括单链表的理解)
思想:
遵循以下原则
1.当前结点cur没有左子树,当前结点cur向右移动
2.当前结点cur有左子树,找到当前结点左子树的最右结点
如果左子树的最右结点的右子树为空,那么将本来指向空的最右结点指向当前结点cur,然后当前结点向左移动
如果当前结点cur左子树的最右结点指向当前结点cur,让最右孩子指向空,cur向右移动。
遍历顺序如下图
总结:在树中,对于任何一个有左子树的结点会两次回到自己,对于没有左子树的结点,只会到达结点一次,顺序是先遍历当前结点,然后左子树,然后回到当前结点,然后右子树。当前结点一定是我左子树最右孩子的后继结点。
代码:
1 1 public static void morrisIn(Node head) { 2 2 if (head == null) { 3 3 return; 4 4 } 5 5 //当前结点 6 6 Node cur1 = head; 7 7 Node cur2 = null; 8 8 while (cur1 != null) { 9 9 //当前结点的左孩子 10 10 cur2 = cur1.left; 11 11 if (cur2 != null) { 12 12 //当左孩子的右子树不是空切不是cur1时一直向右移动 13 13 while (cur2.right != null && cur2.right != cur1) { 14 14 cur2 = cur2.right; 15 15 } 16 16 //cur2最右子树为空,就让它指向cur1,然后cur1左移 17 17 if (cur2.right == null) { 18 18 cur2.right = cur1; 19 19 cur1 = cur1.left; 20 20 //跳入大循环的下一次 21 21 continue; 22 22 } else { 23 23 cur2.right = null; 24 24 } 25 25 } 26 26 //包含两种情况,一是没左子树直接右移,二是第二次到达当前结点(在执行完cur2.right=null) 27 27 cur1 = cur1.right; 28 28 } 29 29 System.out.println(); 30 30
由Morris变形的二叉树遍历
先序:
#######思想:
我们观察Morris遍历的结果和分析可知,Morris遍历其实就是不加打印的二叉树遍历,把打印语句插入到里面就可以改写成先序中序或者后序
改成先序,即第一次到达当前结点的时候打印,第二次到达不打印,而在代码中怎么体现呢,当左子树最右结点第一次指向cur时,说明cur结点第一次到达,就打印,或者当前结点的左子树为空,也代表了第一次到达且只能到达一次打印,就在代码里加上两个打印语句即可
#######代码:
1 public static void morrisPre(Node head) { 2 if (head == null) { 3 return; 4 } 5 Node cur1 = head; 6 Node cur2 = null; 7 while (cur1 != null) { 8 cur2 = cur1.left; 9 if (cur2 != null) { 10 while (cur2.right != null && cur2.right != cur1) { 11 cur2 = cur2.right; 12 } 13 if (cur2.right == null) { 14 cur2.right = cur1; 15 //代表第一次到达打印 16 System.out.print(cur1.value + " "); 17 cur1 = cur1.left; 18 continue; 19 } else { 20 cur2.right = null; 21 } 22 } else { 23 //代表没有左子树打印 24 System.out.print(cur1.value + " "); 25 } 26 cur1 = cur1.right; 27 } 28 System.out.println(); 29 }
中序:
#######思想:
中序即第二次到达的时候再打印或者没有左子树直接打印,我们经过分析,什么时候满足以上两种情况呢,就是当cur要右移之前,每当cur右移都包含了第二次到达或者没有左子树,所以就在cur右移的代码前加上打印语句即可
#######代码:
1 public static void morrisIn(Node head) { 2 if (head == null) { 3 return; 4 } 5 //当前结点 6 Node cur1 = head; 7 Node cur2 = null; 8 while (cur1 != null) { 9 //当前结点的左孩子 10 cur2 = cur1.left; 11 if (cur2 != null) { 12 13 while (cur2.right != null && cur2.right != cur1) { 14 cur2 = cur2.right; 15 } 16 17 if (cur2.right == null) { 18 cur2.right = cur1; 19 cur1 = cur1.left; 20 //跳入大循环的下一次 21 continue; 22 } else { 23 cur2.right = null; 24 } 25 } 26 //这里!!!!! 27 System.out.print(cur1.value + " "); 28 cur1 = cur1.right; 29 } 30 System.out.println(); 31 }
后序:
#######思想:
基本思路是逆序打印第二次到达结点的右边界,第二次到达结点的打印完了之后,逆序打印整个二叉树的有边界
有边界就是,当前结点的左子树的所有右结点一直到头如下图
注意在逆序的时候,第一轮循环只是改变结点,没改变指向,到了下一轮循环才改变指针的方向
单链表操作 如果是比如from.right = pre 这就表明的是from指向pre 你可以看成 from->right=pre通俗理解为from往右指是pre,然后比如from=next 就相当于,from指向next指向的结点,就相当于from和next指向同一个结点当a.next在等式的左边时,比如a.next=b,就理解为a的下一个节点是b,当a.next结点在右边时比如,b=a.next,就相当于b指向了a.next的位置
#######代码:
1 public static void morrisPos(Node head) { 2 if (head == null) { 3 return; 4 } 5 Node cur1 = head; 6 Node cur2 = null; 7 while (cur1 != null) { 8 cur2 = cur1.left; 9 if (cur2 != null) { 10 while (cur2.right != null && cur2.right != cur1) { 11 cur2 = cur2.right; 12 } 13 if (cur2.right == null) { 14 cur2.right = cur1; 15 cur1 = cur1.left; 16 continue; 17 } else { 18 cur2.right = null; 19 printEdge(cur1.left); 20 } 21 } 22 cur1 = cur1.right; 23 } 24 printEdge(head); 25 System.out.println(); 26 } 27 28 public static void printEdge(Node head) { 29 Node tail = reverseEdge(head); 30 Node cur = tail; 31 while (cur != null) { 32 System.out.print(cur.value + " "); 33 cur = cur.right; 34 } 35 reverseEdge(tail); 36 } 37 38 public static Node reverseEdge(Node from) { 39 Node pre = null; 40 Node next = null; 41 while (from != null) { 42 next = from.right; 43 from.right = pre; 44 pre = from; 45 from = next; 46 } 47 return pre; 48 }

浙公网安备 33010602011771号