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实现二叉树的深度优先遍历(前序、中序、后序)和广度优先遍历(层次)...

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

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

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • css 半透明滚动条「建议收藏」

    css 半透明滚动条「建议收藏」::-webkit-scrollbar{width:10px;height:10px;}::-webkit-scrollbar-thumb{background:hsl(0,0%,51%);-webkit-box-shadow:none;border-radius:10px;-webkit-box-shadow:none;}::-webki…

  • ORACLE恢复数据

    ORACLE恢复数据ORACLE恢复删除表或表记录一:表的恢复对误删的表,只要没有使用PURGE永久删除选项,那么从flashback区恢复回来希望是挺大的。一般步骤有:1、从flashback里查询被删除的表

  • matlab之BM3D

    matlab之BM3D1.matlab代码2.

  • 免费且超级好用的搜索引擎INSO[通俗易懂]

    免费且超级好用的搜索引擎INSO[通俗易懂]免费且超级好用的搜索引擎INSO已经上线啦,界面UI是采用FlatUI设计,能够搜索到很多很多资源,近期资源一般来说要等10天左右,否则基本上是枪版。后面我会推出开发这个搜索引擎的系列教程的,尽请期待!网址是http://www.inso.hk

  • XSHELL安装指南

    XSHELL安装指南开发环境部署目的:利用ssh远程登陆服务器(在windows系统下远程连接linux)下载XSHELL7XSHELL7下载网址:https://www.netsarang.com/zh/xshell/点击“下载”点击“免费授权界面”以上是XSHELL7的下载过程然后找到右键“以管理员身份运行”一上来会出现这种错误,先点击“是(Y)”过程中一直点击“下一步”,以及“我同意”类似的,然后选择个安装路径就可以没啥特殊的。到最后一切顺利的话会显示下面这样的界面一般通向成功的道

    2022年10月31日
  • gbase导出sql_gbase修改字段名称

    gbase导出sql_gbase修改字段名称喵了个咪的。到目前为止,自己已经用过SQLSERVER,MySQL,Oracle,SQLite,加上南大通用GBASE五种数据库了。虽然每种都用的不深注:GBASE提供了C的API,查看手册即可。不支持string。用C++配置GBASE:对方提供了32位和64位windows下的库。在程序中添加gbase.herrmsg.h两个头文件,导入gbaseclient.liblibgb…

    2022年10月31日

发表回复

您的电子邮箱地址不会被公开。

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