PHP实现二叉树的深度优先遍历(前序、中序、后序)和广度优先遍历(层次)…

PHP实现二叉树的深度优先遍历(前序、中序、后序)和广度优先遍历(层次)…

大家好,又见面了,我是全栈君。

点击上方“码农编程进阶笔记”,选择“置顶或者星标

优质文章第一时间送达!

前言:

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

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

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

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

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

例如对于一下这棵树:

PHP实现二叉树的深度优先遍历(前序、中序、后序)和广度优先遍历(层次)...

深度优先遍历:

前序遍历: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);
    }

说明: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);
    }




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);
    }

广度优先遍历:

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;
    }

说明: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();

PHP实现二叉树的深度优先遍历(前序、中序、后序)和广度优先遍历(层次)...

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/111998.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 防火墙透明模式和路由模式区别_防火墙的部署模式

    防火墙透明模式和路由模式区别_防火墙的部署模式防火墙能够工作在三种模式下:路由模式、透明模式、混合模式。如果防火墙以第三层对外连接(接口具有IP地址),则认为防火墙工作在路由模式下;若防火墙通过第二层对外连接(接口无IP地址),则防火墙工作在透明模式下;若防火墙同时具有工作在路由模式和透明模式的接口(某些接口具有IP地址,某些接口无IP地址),则防火墙工作在混合模式下。防火墙三种工作模式的简介1、路由模式当防火墙位于内部网络和外部网络之间时,需要将防火墙与内部网络、外部网络以及DMZ三个区域相连的接口分别配置成不同网段的IP地址

    2022年10月23日
    0
  • java堆栈详解

    java堆栈详解java虚拟机栈栈是线程私有,他的生命周期和线程的相同。用于存储局部变量,操作数栈,动态链接,方法出口等。他会抛出两种异常,stackoverflowerror异常和outofmemoryerror异常。java虚拟机堆堆是线程共有的一块内存区域,在虚拟机启动时创建,为了存放对象实例。java堆是垃圾收集器管理的主要区域,因此很多时候被称为“GC堆”。java堆可以处于物理上不连续的内

    2022年7月8日
    20
  • spss数据分析聚类分析_SPSS聚类分析

    spss数据分析聚类分析_SPSS聚类分析SPSS之聚类分析(图文+数据集)聚类分析简介按照个体(记录)的特征将它们分类,使同一类别内的个体具有尽可能高的同质性,而类别之间则具有尽可能高的异质性。为了得到比较合理的分类,首先要采用适当的指标来定量地描述研究对象之间的联系的紧密程度。假定研究对象均用所谓的“点”来表示。在聚类分析中,一般的规则是将“距离”较小的点归为同一类,将“距离”较大的点归为不…

    2022年10月17日
    0
  • PKI体系和数字证书[通俗易懂]

    PKI体系和数字证书[通俗易懂]PKI体系【通过使用公钥技术和数字签名来确定信息安全,并负责验证数字证书持有者的身份的一种技术】PKI的组成?公钥加密技术、数字证书、CA(授权机构)、RA(注册机构)数据加密的过程是?(对称加密)a.发送方A用接收方B的公钥加密数据b.接收方B用自己的私钥解密数据数字签名的过程是?(非对称加密)a.被发送文件被用某种HASH算法产生“数字摘要”;b.发送方用自己的私钥对“数字摘要”进行加密,形成数字签名;c.将源文件和加密的摘要同时传给对方;d.接收方用发送方的公钥对摘要进行解密、同

    2022年8月22日
    4
  • x201换风扇_笔记本怎么换风扇 ThinkPad X201i换风扇图文教程

    x201换风扇_笔记本怎么换风扇 ThinkPad X201i换风扇图文教程ThinkPadX201i换电扇图文教程:拆机之前,我们需求先对X201i的散热电扇在停止了开端的理解,得知价钱从10元左右的单电扇,到上百的散热全体都有,而且还分东芝产和松下产等不同产地的,小编选择了松下产的整套散热(包括散热片和电扇),价钱为150,电扇固定办法为小螺丝。假定拿到电脑修理店去换的话,小编猜测我们所需求的费用至少在200-300元之间。一:拆机前的准备螺丝刀,小毛刷和安排螺丝的…

    2022年6月27日
    44
  • Android StrictMode 详解

    Android StrictMode 详解Android2.3提供一个称为严苛模式(StrictMode)的调试特性,Google称该特性已经使数百个Android上的Google应用程序受益。它将报告与线程及虚拟机相关的策略违例。一旦检测到策略违例(policyviolation),将获得警告,其包含了一个栈trace显示你的应用在何处发生违例。可以强制用警告代替崩溃(crash),也可以仅将警告计入日志,让你的应用继续执行St

    2022年6月1日
    36

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号