php-二叉树

二叉树的基本知识就不说了,谈一些有意思的问题!

想要存储一棵二叉树,我们有两种方法

一种是基于指针或者引用的二叉链式存储法,
一种是基于数组的顺序存储法。

链式存储 大部分二叉树代码都是通过这种结构来实现的。

链式存储

每个节点有 3 个字段,data 存储数据,另外两个是指向左右子节点的指针

顺序存储 存储适用性不强,一般只用于完全二叉树

顺序存储

二叉树的遍历

二叉树的遍历方式有很多,如果限制了从左到右的习惯方式,那么主要就分为四种:

  • 前序遍历(DLR):根结点,遍历左子树,遍历右子树【根左右】百度百科

    前序遍历: ABDGHCEIF

  • 中序遍历(LDR):遍历左子树,根结点,遍历右子树【左根右】百度百科

    中序遍历: GDHBAEICF

  • 后序遍历(LRD):遍历左子树,遍历右子树,根结点【左右根】百度百科

    后序遍历:GHDBIEFCA

  • 层序遍历:从根结点,从上而下逐层遍历


    层序遍历:ABCDEFGHI

那为什么要有这么多遍历方法呢

我们用图形的方式来表现树的结构,可以非常直观的理解,但是对于计算机来说,它只有循环、判断等方式来处理,换种方式说,就是它只会处理线性序列,而刚才四种遍历方法,都是把树中的结点变成某种意义的线性结构。

推导二叉树

很多面试会问这个问题。给出前序ABCDEF,中序CBAEDF,问后序是什么?
其实很简单前序先访问根结点 A,中序 A 左边就是左子树,可以根据上面的图自己推导着画一下,结果CBEFDA

简单的二叉树代码实现

class node
{
    public $data   = null;
    public $left   = null;
    public $right  = null;
    public $parent = null;

    function __construct($data)
    {
        $this->data = $data;
    }
}

class tree
{
    public $root  = null;
    public $size  = 0;
    public $depth = null;

    function __construct($data)
    {
        $this->root = new node($data);
        $this->size++;
        $this->depth++;
    }

    function addNode($arr)
    {
        foreach ($arr as $value) {
            $current     = $this->root;
            $parent      = null;
            $currentDept = 1;

            while ($current !== null) {
                $parent = $current;
                if ($value == $current->data) {
                    continue 2;
                } elseif ($current->data > $value) {
                    $current = $current->left;
                } else {
                    $current = $current->right;
                }
                $currentDept++;
            }
            $node         = new node($value);
            $node->parent = $parent;
            if ($parent->data > $value) {
                $parent->left = $node;
            } else {
                $parent->right = $node;
            }

            if ($this->depth < $currentDept) {
                $this->depth++;
            }
            $this->size++;
        }

        return true;
    }

    function showTree()
    {
        return $this->root;
    }
}

二叉查找树

查找树其实就是在规则上限制了数据。它不仅仅支持快速查找一个数据,还支持快速插入、删除一个数据。

它是怎么做到这些的呢?

二叉查找树要求,在树中的任意一个节点,其左子树中的每个节点的值,都要小于这个节点的值,而右子树节点的值都大于这个节点的值。

时间复杂度其实都跟树的高度成正比 O(logn)

散列表和二叉查找树

散列表的插入、删除、查找操作的时间复杂度可以做到常量级的 O(1)
二叉查找树在比较平衡的情况下,插入、删除、查找操作时间复杂度才是 O(logn)
相对散列表,好像并没有什么优势,那我们为什么还要用二叉查找树呢?

  • 如果要输出有序数据,散列表就要先排序,二叉树则可以通过中序遍历在 O(n) 时间复杂度内输出
  • 散列表扩容耗时很多,而且当遇到散列冲突时,性能不稳定,尽管二叉查找树的性能不稳定,但我们最常用的平衡二叉查找树的性能非常稳定,时间复杂度稳定在 O(logn)。
  • 尽管散列表的查找等操作的时间复杂度是常量级的,平衡二叉查找树是 O(logn) 但因为哈希冲突的存在,这个常量不一定比 logn 小,所以实际的查找速度可能不一定快
  • 散列表的构造比二叉查找树要复杂,需要考虑的东西很多。比如散列函数的设计、冲突解决办法、扩容、缩容等。平衡二叉查找树只需要考虑平衡性这一个问题,而且这个问题的解决方案比较成熟、固定。
  • 为了避免过多的散列冲突,散列表装载因子不能太大,特别是基于开放寻址法解决冲突的散列表,不然会浪费一定的存储空间。

平衡二叉查找树,红黑树?红黑树

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 228,505评论 6 533
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 98,556评论 3 418
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 176,463评论 0 376
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 63,009评论 1 312
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 71,778评论 6 410
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 55,218评论 1 324
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 43,281评论 3 441
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 42,436评论 0 288
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 48,969评论 1 335
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 40,795评论 3 354
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 42,993评论 1 369
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 38,537评论 5 359
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 44,229评论 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 34,659评论 0 26
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 35,917评论 1 286
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 51,687评论 3 392
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 47,990评论 2 374