(310)排序-二叉排序树

概述

  二叉排序树又称“二叉查找树”、“二叉搜索树”。二叉排序树:或者是一棵空树,或者是具有下列性质的二叉树:

  1. 若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值;
  2. 若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值;
  3. 它的左、右子树也分别为二叉排序树。

理论参考

http://www.cnblogs.com/zhuyf87/archive/2012/11/09/2763113.html
http://blog.chinaunix.net/uid-22663647-id-1771796.html
http://blog.csdn.net/hackbuteer1/article/details/6583988

代码(PHP)

1.二叉树类

在这个类中完成基本二叉树定义,包括定义属性(当前节点值,左子树,右子树)和一些基本方法(先序遍历,中序遍历,后序遍历)。

//定义一个二叉树
class BTree{
    protected $key   = null;//当前节点值
    protected $left  = null;//左子树
    protected $right = null;//右子树

    //构造函数
    public function __construct($i_key=null,$i_left=null,$i_right=null){
        $this->key = $i_key;
        $this->left = $i_left;
        $this->right = $i_right;
    }
    //析构函数
    public function __destruct(){
        $this->key = null;
        $this->left = null;
        $this->right = null;
    }

    //先序遍历,非递归实现
    public function preOrderTraversal(){
        $arr = array();
        $stack=array();
        $temp_tree = $this;
        while($temp_tree != null){
            $arr[] = $temp_tree->key;
            array_push($stack,$temp_tree);
            $temp_tree = $temp_tree->left;
        }
        while(!empty($stack)){
            $temp_tree = array_pop($stack);
            $temp_tree = $temp_tree->right;
            while($temp_tree != null){
                $arr[] = $temp_tree->key;
                array_push($stack,$temp_tree);
                $temp_tree = $temp_tree->left;
            }
        }
        return $arr;
    }

    //中序遍历,非递归实现
    public function inOrderTraversal(){
        $arr= array();  //存放遍历结果
        $stack = array();//存放节点栈
        $temp_tree =$this;
        while($temp_tree != null || !empty($stack)){
            while($temp_tree != null){
                array_push($stack,$temp_tree);
                $temp_tree = $temp_tree->left;
            }
            if(!empty($stack)){
                $temp_tree = array_pop($stack);
                $arr[]=$temp_tree->key;
                $temp_tree = $temp_tree->right;
            }
        }
        return $arr;
    }

    //后续遍历,非递归实现
    public function postOrderTraversal(){
        $arr= array();  //存放遍历结果
        $stack = array();//存放节点栈
        $temp_tree =$this;
        $previsit =null;
        while($temp_tree != null || !empty($stack)){
            while($temp_tree !=null){
                array_push($stack,$temp_tree);
                $temp_tree =$temp_tree->left;
            }

            $temp_tree = array_pop($stack);
            if($temp_tree->right == null || $temp_tree->right == $previsit){
                $arr[] = $temp_tree->key;
                $previsit = $temp_tree;
                $temp_tree = null;
            }else{
                array_push($stack,$temp_tree);
                $temp_tree = $temp_tree->right;
            }
        }
        return $arr;
    }
}

2.二叉排序树类

该类继承与二叉树类(BTree),完成二叉排序树的插入节点,查找节点,删除节点。

//定义二叉排序树
class BinarySortTree extends BTree{
    //插入一个节点到当前树中
    public function insertNode($key){
        if($this->key == null){//如果是空树,插入到首节点。
            $this->key = $key;
            return;
        }
        $temp_tree =$this;//当前子树
        while($temp_tree !=null){
            if($temp_tree->key == $key){
                break;
            }
            if($temp_tree->key > $key){//左子树
                if($temp_tree->left == null){
                    $temp_tree->left = new BinarySortTree($key);
                    break;
                }else{
                    $temp_tree = $temp_tree->left;
                }
            }
            if($temp_tree->key < $key){//右子树插入
                if($temp_tree->right == null){
                    $temp_tree->right = new BinarySortTree($key);
                    break;
                }else{
                    $temp_tree = $temp_tree->right;
                }
            }
        }
    }

    //查找一个节点,找到返回该节点及其子树,否则返回null
    public function searchNode($key){
        $temp_tree =$this;//当前子树
        while($temp_tree !=null){
            if($temp_tree->key == $key){
                break;
            }
            if($temp_tree->key > $key){//左子树
                if($temp_tree->left == null){
                    $temp_tree = null;
                    break;
                }else{
                    $temp_tree = $temp_tree->left;
                }
            }
            if($temp_tree->key < $key){//右子树插入
                if($temp_tree->right == null){
                    $temp_tree=null;
                    break;
                }else{
                    $temp_tree = $temp_tree->right;
                }
            }
        }
        return $temp_tree;
    }

    //删除一个节点
    public function deleteNode($key){
        $parent_tree =null;//要删除节点的父节点树
        $temp_tree = $this;//要删除的节点
        $in_side = 0;       //要删除的节点在父节点树的哪边

        //找到要删除的节点极其父节点
        while($temp_tree !=null && ($temp_tree->key != $key)){
            if($temp_tree->key > $key){//左子树
                if($temp_tree->left == null){
                    $temp_tree = null;
                    break;
                }else{
                    $in_side =0;
                    $parent_tree = $temp_tree;
                    $temp_tree = $temp_tree->left;
                }
            }else{//右子树
                if($temp_tree->right == null){
                    $temp_tree=null;
                    break;
                }else{
                    $in_side =1;
                    $parent_tree = $temp_tree;
                    $temp_tree = $temp_tree->right;
                }
            }
        }
        //根据不同情况进行删除操作
        if($temp_tree != null){//当前节点存在
            $p_side =null;
            //开始删除
            if($temp_tree->left == null){
                //如果要删除节点左边为空,就将右边赋给parent;
                $p_side = $temp_tree->right;
            }else if($temp_tree->right == null){
                //如果要删除节点右边边为空,就将左边赋给parent;
                $p_side = $temp_tree->left;
            }else{
                //都不为空,找到要删除节点左子树的最大的节点,极其该节点的父节点
                $lMax =$temp_tree->right;//左子树最大节点
                $p_lMax = $temp_tree;
                while($lMax->right != null){
                    if($lMax->right->right == null){
                        $p_lMax = $lMax;
                    }
                    $lMax = $lMax->right;
                }
                $p_lMax->right = $lMax->left;
                $lMax->left = $temp_tree->left;
                $lMax->right = $temp_tree->right;
                $p_side = $lMax;
            }
            //设置父节点
            if($parent_tree != null){//不是根节点删除
                //引用当前父节点的某一边。
                if($in_side == 0){
                    $parent_tree->left  = $p_side;
                }else{
                    $parent_tree->right = $p_side;
                }
            }else{
                $this->key = $p_side->key;
                $this->left = $p_side->left;
                $this->right = $p_side->right;
            }
        }
    }
}

3.调用

$item = array(50, 30, 20,35,33,40,36, 100, 56, 78);
$root = new BinarySortTree();
foreach($item as $key){
    $root->insertNode($key);
}
var_dump($root);
echo '先序遍历:'.implode(',',$root->preOrderTraversal()).'<br>';
echo '中序遍历:'.implode(',',$root->inOrderTraversal()).'<br>';
echo '后序遍历:'.implode(',',$root->postOrderTraversal()).'<br>';

$root->deleteNode('30');
echo '删除节点后的先序遍历:'.implode(',',$root->preOrderTraversal()).'<br>';

结果

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

推荐阅读更多精彩内容

  • 树的概述 树是一种非常常用的数据结构,树与前面介绍的线性表,栈,队列等线性结构不同,树是一种非线性结构 1.树的定...
    Jack921阅读 4,432评论 1 31
  • 四、树与二叉树 1. 二叉树的顺序存储结构 二叉树的顺序存储就是用数组存储二叉树。二叉树的每个结点在顺序存储中都有...
    MinoyJet阅读 1,499评论 0 7
  • B树的定义 一棵m阶的B树满足下列条件: 树中每个结点至多有m个孩子。 除根结点和叶子结点外,其它每个结点至少有m...
    文档随手记阅读 13,161评论 0 25
  • 二叉树 满二叉树 国内教程定义:一个二叉树,如果每一个层的结点数都达到最大值,则这个二叉树就是满二叉树。也就是说,...
    简_爱SimpleLove阅读 4,249评论 0 3
  • 姓名: 李小娜 [嵌牛导读] :这篇文章主要介绍了Java二叉排序树,包括二叉排序树的定义、二叉排序树的性质、二叉...
    n184阅读 617评论 0 0