http://blog.csdn.net/baidu_30000217/article/details/52953127

前言:

深度优先遍历:对每一个可能的分支路径深入到不能再深入为止,而且每个结点只能访问一次。要特别注意的是,二叉树的深度优先遍历比较特殊,可以细分为先序遍历、中序遍历、后序遍历。具体说明如下:

  • 前序遍历:根节点->左子树->右子树

  • 中序遍历:左子树->根节点->右子树

  • 后序遍历:左子树->右子树->根节点

广度优先遍历:又叫层次遍历,从上往下对每一层依次访问,在每一层中,从左往右(也可以从右往左)访问结点,访问完一层就进入下一层,直到没有结点可以访问为止。

例如对于一下这棵树:

深度优先遍历:

  • 前序遍历:10 8 7 9 12 11 13

  • 中序遍历:7 8 9 10 11 12 13

  • 后序遍历:7 9 8 11 13 12 10

广度优先遍历:

  • 层次遍历:10 8 12 7 9 11 13

二叉树的深度优先遍历的非递归的通用做法是采用栈,广度优先遍历的非递归的通用做法是采用队列。

深度优先遍历:

1、前序遍历:

    /**      * 前序遍历(递归方法)      */     private function pre_order1($root)     {        if (!is_null($root)) {            //这里用到常量__FUNCTION__,获取当前函数名,好处是假如修改函数名的时候,里面的实现不用修改             $function = __FUNCTION__;             echo $root->key . " ";            $this->$function($root->left);            $this->$function($root->right);         }     }    /**      * 前序遍历(非递归方法)      * 因为当遍历过根节点之后还要回来,所以必须将其存起来。考虑到后进先出的特点,选用栈存储。      */     private function pre_order2($root)     {        //        $stack = new splstack();         //        $stack->push($root);         //        while(!$stack->isEmpty()){         //            $node = $stack->pop();         //            echo $node->key.' ';         //            if(!is_null($node->right)){         //                $stack->push($node->right);         //            }         //            if(!is_null($node->left)){         //                $stack->push($node->left);         //            }         //        }         if (is_null($root)) {            return;         }        $stack = new splstack();        $node = $root;        while (!is_null($node) || !$stack->isEmpty()) {            while (!is_null($node)) {                //只要结点不为空就应该入栈保存,与其左右结点无关                 $stack->push($node);                 echo $node->key . ' ';                $node = $node->left;             }            $node = $stack->pop();            $node = $node->right;         }     }    //前序遍历     public function PreOrder()     {        // 所在对象中的tree属性保存了一个树的引用         //     $this->pre_order1($this->tree->root);         $this->pre_order2($this->tree->root);     }123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960

说明:1、我将所有的遍历方法都封装在一个类 traverse 里面了。2、pre_order2方法中,在使用栈的过程中,我使用的是PHP标准库SPL提供的splstack,如果你们习惯使用数组的话,可以使用 array_push() 和array_pop() 模拟实现。

2、中序遍历:

    /**      * 中序遍历(递归方法)      */     private function mid_order1($root)     {        if (!is_null($root)) {            $function = __FUNCTION__;            $this->$function($root->left);             echo $root->key . " ";            $this->$function($root->right);         }     }    /**      * 中序遍历(非递归方法)      * 因为当遍历过根节点之后还要回来,所以必须将其存起来。考虑到后进先出的特点,选用栈存储。      */     private function mid_order2($root)     {        if (is_null($root)) {            return;         }        $stack = new splstack();        $node = $root;        while (!is_null($node) || !$stack->isEmpty()) {            while (!is_null($node)) {                $stack->push($node);                $node = $node->left;             }            $node = $stack->pop();             echo $node->key . ' ';            $node = $node->right;         }     }    //中序遍历     public function MidOrder()     {        //        $this->mid_order1($this->tree->root);         $this->mid_order2($this->tree->root);     }123456789101112131415161718192021222324252627282930313233343536373839404142123456789101112131415161718192021222324252627282930313233343536373839404142

3、后序遍历:

    /**      * 后序遍历(递归方法)      */     private function post_order1($root)     {        if (!is_null($root)) {            $function = __FUNCTION__;            $this->$function($root->left);            $this->$function($root->right);             echo $root->key . " ";         }     }    /**      * 后序遍历(非递归方法)      * 因为当遍历过根节点之后还要回来,所以必须将其存起来。考虑到后进先出的特点,选用栈存储。      * 由于在访问了左子节点后怎么跳到右子节点是难点,这里使用一个标识lastVisited来标识上一次访问的结点      */     private function post_order2($root)     {        if (is_null($root)) {            return;         }        $node = $root;        $stack = new splstack();        //保存上一次访问的结点引用         $lastVisited = NULL;        $stack->push($node);        while(!$stack->isEmpty()){            $node = $stack->top();//获取栈顶元素但不弹出             if(($node->left == NULL && $node->right == NULL) || ($node->right == NULL && $lastVisited == $node->left) || ($lastVisited == $node->right)){                 echo $node->key.' ';                $lastVisited = $node;                $stack->pop();             }else{                if($node->right){                    $stack->push($node->right);                 }                if($node->left){                    $stack->push($node->left);                 }             }         }     }    //后序遍历     public function PostOrder()     {        //        $this->post_order1($this->tree->root);         $this->post_order2($this->tree->root);     }1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515212345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152

广度优先遍历:

1、层次遍历:

    /**      * 层次遍历(递归方法)      * 由于是按层逐层遍历,因此传递树的层数      */     private function level_order1($root,$level){        if($root == NULL || $level < 1){            return;         }        if($level == 1){             echo $root->key.' ';            return;         }        if(!is_null($root->left)){            $this->level_order1($root->left,$level - 1);         }        if(!is_null($root->right)){            $this->level_order1($root->right,$level - 1);         }     }    /**      * 层次遍历(非递归方法)      * 每一层从左向右输出 元素需要储存有先进先出的特性,所以选用队列存储。      */     private function level_order2($root){        if(is_null($root)){            return;         }        $node = $root;        //利用队列实现//        $queue = array();//        array_push($queue,$node);        while(!is_null($node = array_shift($queue))){//            echo $node->key.' ';//            if(!is_null($node->left)){//                array_push($queue,$node->left);//            }//            if(!is_null($node->right)){//                array_push($queue,$node->right);//            }//        }         $queue = new splqueue();        $queue->enqueue($node);        while(!$queue->isEmpty()){            $node = $queue->dequeue();             echo $node->key.' ';            if (!is_null($node->left)) {                $queue->enqueue($node->left);             }            if (!is_null($node->right)) {                $queue->enqueue($node->right);             }         }     }    //层次遍历     public function LevelOrder(){//        $level = $this->getdepth($this->tree->root);//        for($i = 1;$i <= $level;$i ++){//            $this->level_order1($this->tree->root,$i);//        }         $this->level_order2($this->tree->root);     }    //获取树的层数     private function getdepth($root){        if(is_null($root)){            return 0;         }        $left = getdepth($root -> left);        $right = getdepth($root -> right);        $depth = ($left > $right ? $left : $right) + 1;        return $depth;     }1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787912345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879

说明:level_order2方法中,在使用队列的过程中,我使用的是php标准库SPL提供的splqueue,如果你们习惯使用数组的话,可以使用 array_push() 和array_shift() 模拟实现。

使用:

现在我们来看看客户端代码:

class Client{     public static function Main()     {         try {            //实现文件的自动加载             function autoload($class)             {                 include strtolower($class) . '.php';             }             spl_autoload_register('autoload');            $arr = array(10, 8, 12, 7, 9, 11, 13);            $tree = new Bst();//            $tree = new Avl();//            $tree = new Rbt();             $tree->init($arr);            $traverse = new traverse($tree);            $traverse->PreOrder();//            $traverse->MidOrder();//            $traverse->PostOrder();//            $traverse->LevelOrder();         } catch (Exception $e) {            echo $e->getMessage();         }     } } CLient::Main();12345678910111213141516171819202122232425262728293031321234567891011121314151617181920212223242526272829303132

补充: 
1. 在客户端中所使用的三个类 Bst、Avl、Rbt 大家可以参考我的其他三篇博客:《PHP二叉树(一):二叉搜索树》、《 PHP二叉树(二):平衡二叉树(AVL)》、《PHP二叉树(三):红黑树》 
2. 为什么我推荐大家使用SPL标准库中提供的splstack和splqueue呢?这是我在某一篇文章中看到的:虽然我们可以使用传统的变量类型来描述数据结构,例如用数组来描述堆栈(Strack)– 然后使用对应的方式 pop 和 push(array_pop()、array_push()),但你得时刻小心,因为毕竟它们不是专门用于描述数据结构的 – 一次误操作就有可能破坏该堆栈。而 SPL 的 SplStack 对象则严格以堆栈的形式描述数据,并提供对应的方法。同时,这样的代码应该也能理解它在操作堆栈而非某个数组,从而能让你的同伴更好的理解相应的代码,并且它更快。原文地址:PHP SPL,遗落的宝石 
3. 本篇博客有参考: 《数据结构(六)——二叉树 前序、中序、后序、层次遍历及非递归实现 查找、统计个数、比较、求深度的递归实现》、《二叉树的深度优先遍历和广度优先遍历》

本文转自  陈小龙哈   51CTO博客,原文链接:http://blog.51cto.com/chenxiaolong/1959144

PHP实现二叉树的深度优先遍历(前序、中序、后序)和广度优先遍历(层次) 转载陈小龙哈2017...相关推荐

  1. 二叉树的深度优先搜索, 前序中序后序

    深度优先搜索即 depth first search, 简称 DFS 从根节点触发, 沿着路径, 达到最大层次的节点, 然后回退到上一节点, 继续搜索 有前序, 中序, 后序三种方式 前序遍历 根左右 ...

  2. Python__数据结构与算法——树、二叉树(实现先、中、后序遍历)

    目录 一.树 二.二叉树 树和前面所讲的表.堆栈和队列等这些线性数据结构不同,树不是线性的.在处理较多数据时,使用线性结构较慢,而使用树结构则可以提高处理速度.不过,相对于线性的表.堆栈和队列等线性数 ...

  3. 二叉树前序遍历python输出_[宜配屋]听图阁 - Python实现输入二叉树的先序和中序遍历,再输出后序遍历操作示例...

    本文实例讲述了Python实现输入二叉树的先序和中序遍历,再输出后序遍历操作.分享给大家供大家参考,具体如下: 实现一个功能: 输入:一颗二叉树的先序和中序遍历 输出:后续遍历 思想: 先序遍历中,第 ...

  4. 二叉树的前序中序后序遍历

    二叉树的前序中序后序遍历 二叉树的遍历 前序遍历 中序遍历 后序遍历 总结 二叉树的遍历 二叉树的遍历有前序遍历,中序遍历,后序遍历三种. 今天我把二叉树的遍历方法给大家总结一下,也算对我自己学习的一 ...

  5. 【二叉树Java】二叉树遍历前序中序后序遍历的非递归写法

    本文主要介绍二叉树前序中序后序遍历的非递归写法 在探讨如何写出二叉树的前序中序后序遍历代码之前,我们先来明确一个问题,前序中序后序遍历根据什么区分? 二叉树的前序中序后序遍历,是相较根节点说的.最先遍 ...

  6. 二叉树的前序中序后序遍历java代码实现

    1.前序遍历概述 前序遍历(VLR) 是二叉树遍历的一种,也叫做先根遍历.先序遍历.前序周游,可记做根左右.前序遍历首先访问根结点然后遍历左子树,最后遍历右子树. 若二叉树为空则结束返回,否则: (1 ...

  7. 二叉树的前序中序后序遍历真题

    一颗二叉树的前序遍历是ABCDFGHE,后序遍历是BGHFDECA,中序遍历是? A.GHBADFCEB.DGBAFHECC.BADGFHCED.BAGDFHEC 正确答案:C 牛客网来源:https ...

  8. 二叉树遍历(递归实现前序/中序/后序遍历)

    1. 准备工作 我们先定义一棵普通的二叉树,如下图 2. 前序遍历 通过递归进行遍历: 如果二叉树为空,则操作返回: 如果非空,否则从根结点开始,然后遍历左子树,再遍历右子树. 前序遍历的结果是:AB ...

  9. C++用类实现二叉树的创建,前序中序后序遍历(附完整代码)

    C++用类实现二叉树的创建,前序中序后序遍历(附完整代码) 前序.中序.后序遍历 直接上代码 前序.中序.后序遍历 二叉树的遍历分为前序遍历,中序遍历和后序遍历三种遍历方法.前序遍历的顺序为" ...

最新文章

  1. yaml加配置文件后起不来_YAML配置文件管理资源
  2. Java 打包 FatJar 方法小结
  3. Java基础——线程及并发机制
  4. 图灵成立七周年——经典回顾
  5. 【Elasticsearch】ES 慢查询 profile 参数
  6. 2019 ACM/ICPC 全国邀请赛(西安)J And And And (树DP+贡献计算)
  7. 《陪孩子像搭积木一样学编程》,一起来玩Scratch(1)使用Scratch编程的基本流程...
  8. 如何使用数据质量服务和SQL Server集成服务清除数据
  9. 天寒宜早睡,梦醒闻雪声,倒计时83
  10. python 面向对象编程、别人么样用_Python 中的面向对象没有意义
  11. 版图与射频IC实习心得
  12. Mac系统重置PostgreSQL密码
  13. excel 如何超链接到另一个工作薄下查找相同字符数据
  14. 浪潮首创“容器+虚拟机”双引擎超融合一体机,降本增效再迎利器
  15. Hazel引擎学习(七)
  16. gb28181简单实现sip信令服务器(java版基于springboot):四、sip摄像头心跳保活、推流(tcp/udp)和结束推流
  17. 产业洞察 | 频繁入股网络安全企业,国有资本这是要搞大事情
  18. 第2节 网络及IP地址详解
  19. Angular-------持续学习(angular8.0)
  20. java动态图片_java实现gif动画效果(java显示动态图片)

热门文章

  1. boost::fusion::push_front用法的测试程序
  2. ITK:创建一个点集
  3. DCMTK:转换dicom文件编码
  4. VTK:Math之MatrixInverse
  5. VTK:相互作用之MoveAGlyph
  6. VTK:相互作用之DoubleClick
  7. OpenCV SURF检测的实例(附完整代码)
  8. OpenCV Mat-基本图像容器
  9. C++happy number开心数的实现算法(附完整源码)
  10. QML基础类型之var