php-二叉树

2020-09-22  本文已影响0人  淡淡de盐

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

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

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

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

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

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

顺序存储

二叉树的遍历

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

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

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

推导二叉树

很多面试会问这个问题。给出前序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)
相对散列表,好像并没有什么优势,那我们为什么还要用二叉查找树呢?

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

上一篇 下一篇

猜你喜欢

热点阅读