二叉树遍历(五-最终篇)

简介: 二叉树遍历(五-最终篇)

前言


啦啦啦,小嘟又和大家见面啦,今天小嘟要和大家聊得话题是二叉树三种遍历方式在代码方面的异同。

涉及的对象:二叉树

涉及的内容: (1)前序遍历 (2)中序遍历 (3)后序遍历(4)如何形象的理解遍历方式

涉及的代码:递归代码和迭代代码

重点:找到一种框架适合这三种遍历方式

注:文章首发在掘金,都是自己的账号,不是抄袭

正文

  • 首先我们将下边这个二叉树的三种遍历结果写出来:


前序遍历:[1,2,4,5,3,6]


中序遍历:[4,2,5,1,3,6]


后序遍历:[4,5,2,6,3,1]


注:讲几个很形象的例子,你就能很快的手写出这三种遍历顺序(这里的例子借鉴的是一篇很nice的博客,感觉很适合小白,原文链接小嘟就放在底部喽,需要的读者可以去看看,这个博主比小嘟厉害多了

1.前序遍历

  • 前序遍历,你可以理解成从根节点出发,以逆时针方向前进,将所有结点都遍历一遍


前序遍历递归代码

var preorderTraversal = function(root) {
      let res = [];//最后返回的数组,存储的是遍历的序列
      //这里用到了es6中的箭头函数
      const traversal = (root01)=>{  //如果当前结点为null,
          if(root01 == null) return ;//说明这条路径走到尽头了,需要重新找路
          res.push(root01.val);     //将遇到的打印,这里是存储哦 
          traversal(root01.left);   //找左孩子
          traversal(root01.right);  //找右孩子
      }
      traversal(root);
      return res;//返回值
};



小嘟稍微解释一下,看着代码,我们知道,只要遇到元素,就会将它打印(这里是存储),然后进入自己的左子树,一直递归调用下去,等左子树为null了,递归开始一层一层的往上返回,每返回一层,然后接着进入自己的右子树,进入右子树,也是先进入右子树中的左子树,以此类推,等所有的都找完了,递归函数的层级减为1(此时我们可以理解成,我们是从根节点出去的,这个时候又回到根节点啦)。此处我们验证了上边的遍历顺序是正确的,但是,这只是个思想,不能认为遍历顺序一定是这样的,在迭代代码中,就没有9、10这两步。

2.中序遍历

  • 中序遍历,可以这样思考,将所有结点垂直投影在底部,例如,我们建立一个二维坐标系,横轴为x,纵轴为y,将上边的图搬到该坐标系上,在x轴上得投影就是中序遍历的结果,这个应该比一个一个数快的多啦。


  • 看图是不是很直观呢?嘿嘿嘿,学会就行,小嘟很是希望读者看一眼就会,加油!!!
  • 中序遍历递归代码
var inorderTraversal = function(root) {
       let res = [];//要返回的数组
       const  traversal = (root01)=>{//es6中的箭头函数
            if(root01 == null) return ;
            traversal(root01.left);
            res.push(root01.val);
            traversal(root01.right);
        }
        traversal(root);
        return res;
 };
  • 中序遍历顺序基本和前序遍历差不多,不同点是打印的位置放在了中间,这也是为什么将这种遍历方式称为中序遍历。
  • 3.后序遍历


后序遍历的话你可以把这个理解成是一串葡萄,并且我们规定每次摘葡萄只能摘一颗,不能贪心,而且规定从左往右(也就是先遍历左子树再遍历右子树)摘葡萄。现在看下图的摘葡萄过程,是不是觉得后序遍历其实也就是那一回事


后序遍历递归代码

var postorderTraversal = function(root) {
  let res = [];//要返回的数组
  const traversal = (root01)=>{
      if(root01 == null) return;//找到尽头,需要另换路
      traversal(root01.left);//遍历左子树
      traversal(root01.right);//遍历右子树
      res.push(root01.val);//左右子树遍历结束就只剩中间喽!
  }
  traversal(root);
  return res;
};


  • 现在看递归代码,你会发现,它们三者的代码竟然如此的相似,小嘟我都不知所错了,心里开心的想着:那我要是会了一种,那另外两种不也就拿捏勒,嘿嘿嘿。
  • 4.递归代码总结篇


综上知,我们可以发现三者的递归代码相似程度非常高,难怪别人问的时候只问一次(另外两个很好推啊)。


小嘟来说说代码之间的差异


小嘟自信的说:(1)打印的位置是不一样的 ,一个在最前边,一个在中间,一个在最后。

(2)它们的递归调用函数的顺序是一样的(这个请读者画一下,自己也画了一下,不画也可以,你看代码,除了push位置不同其他的都一样)。

小嘟楠楠的说:然后也就没什么了。现在来一个框架,以后直接用就可以啦!

var postorderTraversal = function(root) {
   let res = [];//要返回的数组
   const traversal = (root01)=>{
       if(root01 == null) return;//找到尽头,需要另换路
       //@key1处
       traversal(root01.left);//遍历左子树
       //@key2处
       traversal(root01.right);//遍历右子树
       //@key3处
   }
   traversal(root);
   return res;
 };
  • 前序遍历:在@key1处编写打印或者存储代码
  • 中序遍历:在@key2处编写打印或者存储代码
  • 后序遍历:在@key3处编写打印或者存储代码
  • 5.迭代代码总结篇

前序遍历

var preorderTraversal = function(root) {
      let res = [];//最后返回的数组,存储的是遍历的序列
      let stack = [];//循环过程中遇到的结点信息
      while(root || stack.length){
          while(root){
              res.push(root.val); //将遇到的打印,这里是存储哦             
              stack.push(root);   // 这里你可以理解成,该结点还要记着怎样找到
                                  // 自己的右孩子。
              root = root.left;   //找左孩子
          }
          let node = stack.pop(); //找不到左孩子了,那就要找该结点的右孩子啦
          root = node.right;      //找右孩子
      }
      return res;
};

中序遍历

var inorderTraversal = function(root) {
   let res  = [];//最后要返回的数组
   let stack = [];//我们用来存储遍历过程中遇到的元素
   //root == null 并且 stack.length == 0(循环退出)    
   while(root||stack.length){
        while(root){//当前元素存在就循环
           stack.push(root);//把当前元素入栈,不是入的值,而是当前的整个元素
           root = root.left;//继续找它的左孩子
        }
        //该元素没有左孩子,那么我们就应该打印该元素
        let node  = stack.pop();//返回该元素并出栈
        res.push(node.val);//将该元素的值压入要返回的栈里边
        root = node.right;//遍历该元素的右孩子
                            //因为左边和中间(父结点)已经遍历过了
   }
   return res;//返回最终的结果
};

后序遍历

var postorderTraversal = function(root) {
   let res = [];//最后要返回的数组
   let stack = [];//遍历过程中遇到的元素结点
   while(root || stack.length){
       while(root){
           res.unshift(root.val);//这个是数组的一个方法
                                 //可以理解成头插法,往首部插入
           stack.push(root);
           root = root.right;
       }
       let node  = stack.pop();
       root = node.left;
   }
   return res;
}; 

它们之间的差异小嘟在之前的几篇文章中都一一讲过了,读者迷惑的话欢迎阅读。嘿嘿嘿(小嘟会把链接放在文章的最下边),现在直接上框架

var preorderTraversal = function(root) {
      let res = [];//最后返回的数组,存储的是遍历的序列
      let stack = [];//循环过程中遇到的结点信息
      while(root || stack.length){
          while(root){
              //@key1处、@key2处         
              stack.push(root);   // 这里你可以理解成,该结点还要记着怎样找到
                                  // 自己的右孩子。
              root = root.left;   //找左孩子
          }
          let node = stack.pop(); //找不到左孩子了,那就要找该结点的右孩子啦
          @key3处
          root = node.right;      //找右孩子
      }
      return res;
};

前序遍历:在@key1处编写打印或者存储代码


中序遍历:在@key3处编写打印或者存储代码


后序遍历:在@key2处编写存储代码


注:

(1).这里不能直接打印哦,因为我们是倒着找的结点

(2).需要改代码,第一处,root = root.left改为root = root.right

(3).第二处,root = node.right 改为 root = node.left

结尾


到今天为止,关于二叉树的三种遍历方法就完美收官了,谢谢各位读者阅读。

文章字数较多,小嘟难免会手抖(嘿嘿嘿),so,欢迎各位批评指正。

创作不易,小嘟这个周末把时间都花在上边了(呜呜呜),周日晚上23:32才完成,所以如果读者还觉得不错的话,可以给小嘟点赞、评论、关注支持一波,小嘟会继续努力下去的。

若读者想让小嘟写某一方面的内容,欢迎评论,小嘟会尽力去做的(要是实在不会,那小嘟只能认错啦)。

最后,祝大家看完该文章有所收获,我们下期再见。

溜啦溜啦...

相关文章
|
人工智能 算法
【算法分析与设计】递归与分治策略(二)
【算法分析与设计】递归与分治策略
|
机器学习/深度学习 算法 编译器
【算法分析与设计】递归与分治策略(一)
【算法分析与设计】递归与分治策略
|
索引
【LeetCode】环形链表+结论证明
【LeetCode】环形链表+结论证明
67 0
|
索引
【LeetCode】环形链表II+结论证明
【LeetCode】环形链表II+结论证明
68 0
|
算法 搜索推荐 Windows
【算法分析与设计】递归与分治策略(三)
【算法分析与设计】递归与分治策略
|
算法 UED 容器
【算法入门&二叉树】从先中后序的遍历到用中后序列构造二叉树|如何抵挡递归法该死的魅力(上)
【算法入门&二叉树】从先中后序的遍历到用中后序列构造二叉树|如何抵挡递归法该死的魅力
129 0
|
算法 容器
【算法入门&二叉树】从先中后序的遍历到用中后序列构造二叉树|如何抵挡递归法该死的魅力(下
【算法入门&二叉树】从先中后序的遍历到用中后序列构造二叉树|如何抵挡递归法该死的魅力
94 0
|
Java
红黑树(中)完整建树过程
手撕JAVA(十四)一文中有些地方表述有误,笔者日后在做修改。这里用画图的方式展示两次红黑树的完整建图过程,一次简单,一次复杂,根据建图过程,就可以理解红黑树是如何实现的。 总结的几点为: 1.根节点必为黑色,任何调整动作都无法将根节点染红 2.新插入节点的父节点和uncle节点同为红色,直接染黑父节点和uncle节点,染红祖父节点(如果祖父节点为根节点,就免疫)
81 0
|
存储
图的深度遍历和广度遍历
图的深度遍历和广度遍历
103 0
如何根据二叉树的两种遍历方式重建二叉树(理论篇)
如何根据二叉树的两种遍历方式重建二叉树(理论篇)
130 0

热门文章

最新文章