二叉查找树(一)之 图文解析 和 C语言的实现

 

概要

       本章先对二叉树的相关理论知识进行介绍,然后给出C语言的详细实现。关于二叉树的学习,需要说明的是:它并不难,不仅不难,而且它非常简单。初次接触树的时候,我也觉得它似乎很难;而之所产生这种感觉主要是由于二叉树有一大堆陌生的概念、性质等内容。而当我真正的实现了二叉树再回过头来看它的相关概念和性质的时候,觉得原来它是如此的简单!因此,建议在学习二叉树的时候:先对二叉树基本的概念、性质有个基本了解,遇到难懂的知识点,可以画图来帮助理解;在有个基本的概念之后,再亲自动手实现二叉查找树(这一点至关重要!);最后再回过头来总结一下二叉树的理论知识时,你会发现——它的确很简单!在代码实践中,我以"二叉查找树,而不是单纯的二叉树"为例子进行说明,单纯的二叉树非常简单,实际使用很少。况且掌握了二叉查找树,二叉树也就自然掌握了。

      本篇实现的二叉查找树是C语言版的,后面章节再分别给出C++和Java版本的实现。您可以根据自己熟悉的语言进行实践学习!

      请务必深刻理解、实践并掌握"二叉查找树"!它是后面学习AVL树、伸展树、红黑树等相关树结构的基石!

 

目录
1. 树的介绍
2. 二叉树的介绍
3. 二叉查找树的C实现
4. 二叉查找树的C测试程序

转载请注明出处:http://www.cnblogs.com/skywang12345/p/3576328.html


更多内容数据结构与算法系列 目录 

 

树的介绍

1. 树的定义

树是一种数据结构,它是由n(n>=1)个有限节点组成一个具有层次关系的集合。

二叉查找树(一)之 图文解析 和 C语言的实现

把它叫做“树”是因为它看起来像一棵倒挂的树,也就是说它是根朝上,而叶朝下的。它具有以下的特点:
(01) 每个节点有零个或多个子节点;
(02) 没有父节点的节点称为根节点;
(03) 每一个非根节点有且只有一个父节点;
(04) 除了根节点外,每个子节点可以分为多个不相交的子树。

 

2. 树的基本术语

若一个结点有子树,那么该结点称为子树根的"双亲",子树的根是该结点的"孩子"。有相同双亲的结点互为"兄弟"。一个结点的所有子树上的任何结点都是该结点的后裔。从根结点到某个结点的路径上的所有结点都是该结点的祖先。

结点的度:结点拥有的子树的数目。
叶子:度为零的结点。
分支结点:度不为零的结点。
树的度:树中结点的最大的度。

层次:根结点的层次为1,其余结点的层次等于该结点的双亲结点的层次加1。
树的高度:树中结点的最大层次。
无序树:如果树中结点的各子树之间的次序是不重要的,可以交换位置。
有序树:如果树中结点的各子树之间的次序是重要的, 不可以交换位置。
森林:0个或多个不相交的树组成。对森林加上一个根,森林即成为树;删去根,树即成为森林。

 

二叉树的介绍

1. 二叉树的定义

二叉树是每个节点最多有两个子树的树结构。它有五种基本形态:二叉树可以是空集;根可以有空的左子树或右子树;或者左、右子树皆为空。

二叉查找树(一)之 图文解析 和 C语言的实现

 

2. 二叉树的性质

二叉树有以下几个性质:TODO(上标和下标)
性质1:二叉树第i层上的结点数目最多为 2{i-1} (i≥1)。
性质2:深度为k的二叉树至多有2{k}-1个结点(k≥1)。
性质3:包含n个结点的二叉树的高度至少为log2 (n+1)
性质4:在任意一棵二叉树中,若终端结点的个数为n0,度为2的结点数为n2,则n0=n2+1

 

2.1 性质1:二叉树第i层上的结点数目最多为 2{i-1} (i≥1)

证明:下面用"数学归纳法"进行证明。
        (01) 当i=1时,第i层的节点数目为2{i-1}=2{0}=1。因为第1层上只有一个根结点,所以命题成立。
        (02) 假设当i>1,第i层的节点数目为2{i-1}。这个是根据(01)推断出来的!
               下面根据这个假设,推断出"第(i+1)层的节点数目为2{i}"即可。
                由于二叉树的每个结点至多有两个孩子,故"第(i+1)层上的结点数目" 最多是 "第i层的结点数目的2倍"。即,第(i+1)层上的结点数目最大值=2×2{i-1}=2{i}
                故假设成立,原命题得证!

 

2.2 性质2:深度为k的二叉树至多有2{k}-1个结点(k≥1)

证明:在具有相同深度的二叉树中,当每一层都含有最大结点数时,其树中结点数最多。利用"性质1"可知,深度为k的二叉树的结点数至多为:
           20+21+…+2k-1=2k-1
           故原命题得证!

 

2.3 性质3:包含n个结点的二叉树的高度至少为log2 (n+1)

证明:根据"性质2"可知,高度为h的二叉树最多有2{h}–1个结点。反之,对于包含n个节点的二叉树的高度至少为log2(n+1)。

 

2.4 性质4:在任意一棵二叉树中,若终端结点的个数为n0,度为2的结点数为n2,则n0=n2+1

证明:因为二叉树中所有结点的度数均不大于2,所以结点总数(记为n)="0度结点数(n0)" + "1度结点数(n1)" + "2度结点数(n2)"。由此,得到等式一。
         (等式一) n=n0+n1+n2
      另一方面,0度结点没有孩子,1度结点有一个孩子,2度结点有两个孩子,故二叉树中孩子结点总数是:n1+2n2。此外,只有根不是任何结点的孩子。故二叉树中的结点总数又可表示为等式二。
         (等式二) n=n1+2n2+1
        由(等式一)和(等式二)计算得到:n0=n2+1。原命题得证!

 

3. 满二叉树,完全二叉树和二叉查找树

3.1 满二叉树

定义:高度为h,并且由2{h} –1个结点的二叉树,被称为满二叉树。

二叉查找树(一)之 图文解析 和 C语言的实现

 

3.2 完全二叉树

定义:一棵二叉树中,只有最下面两层结点的度可以小于2,并且最下一层的叶结点集中在靠左的若干位置上。这样的二叉树称为完全二叉树。
特点:叶子结点只能出现在最下层和次下层,且最下层的叶子结点集中在树的左部。显然,一棵满二叉树必定是一棵完全二叉树,而完全二叉树未必是满二叉树。

二叉查找树(一)之 图文解析 和 C语言的实现

 

3.3 二叉查找树

定义:二叉查找树(Binary Search Tree),又被称为二叉搜索树。设x为二叉查找树中的一个结点,x节点包含关键字key,节点x的key值记为key[x]。如果y是x的左子树中的一个结点,则key[y] <= key[x];如果y是x的右子树的一个结点,则key[y] >= key[x]。

二叉查找树(一)之 图文解析 和 C语言的实现

在二叉查找树中:
(01) 若任意节点的左子树不空,则左子树上所有结点的值均小于它的根结点的值;
(02) 任意节点的右子树不空,则右子树上所有结点的值均大于它的根结点的值;
(03) 任意节点的左、右子树也分别为二叉查找树。
(04) 没有键值相等的节点(no duplicate nodes)。

在实际应用中,二叉查找树的使用比较多。下面,用C语言实现二叉查找树。

 

二叉查找树的C实现

1. 节点定义

1.1 节点定义

二叉查找树(一)之 图文解析 和 C语言的实现
typedef int Type;

typedef struct BSTreeNode{
    Type   key;                    // 关键字(键值)
    struct BSTreeNode *left;    // 左孩子
    struct BSTreeNode *right;    // 右孩子
    struct BSTreeNode *parent;    // 父结点
}Node, *BSTree;
二叉查找树(一)之 图文解析 和 C语言的实现

二叉查找树的节点包含的基本信息:
(01) key      -- 它是关键字,是用来对二叉查找树的节点进行排序的。
(02) left       -- 它指向当前节点的左孩子。
(03) right    -- 它指向当前节点的右孩子。
(04) parent -- 它指向当前节点的父结点。

 

1.2 创建节点

创建节点的代码

二叉查找树(一)之 图文解析 和 C语言的实现
static Node* create_bstree_node(Type key, Node *parent, Node *left, Node* right)
{
    Node* p;

    if ((p = (Node *)malloc(sizeof(Node))) == NULL)
        return NULL;
    p->key = key;
    p->left = left;
    p->right = right;
    p->parent = parent;

    return p;
}
二叉查找树(一)之 图文解析 和 C语言的实现

 

2 遍历

这里讲解前序遍历中序遍历后序遍历3种方式。

2.1 前序遍历

若二叉树非空,则执行以下操作:
(01) 访问根结点;
(02) 先序遍历左子树;
(03) 先序遍历右子树。

前序遍历代码

二叉查找树(一)之 图文解析 和 C语言的实现
void preorder_bstree(BSTree tree)
{
    if(tree != NULL)
    {
        printf("%d ", tree->key);
        preorder_bstree(tree->left);
        preorder_bstree(tree->right);
    }
}
二叉查找树(一)之 图文解析 和 C语言的实现

 

2.2 中序遍历

若二叉树非空,则执行以下操作:
(01) 中序遍历左子树;
(02) 访问根结点;
(03) 中序遍历右子树。

中序遍历代码

二叉查找树(一)之 图文解析 和 C语言的实现
void inorder_bstree(BSTree tree)
{
    if(tree != NULL)
    {
        inorder_bstree(tree->left);
        printf("%d ", tree->key);
        inorder_bstree(tree->right);
    }
}
二叉查找树(一)之 图文解析 和 C语言的实现

 

2.3 后序遍历

若二叉树非空,则执行以下操作:
(01) 后序遍历左子树;
(02) 后序遍历右子树;
(03) 访问根结点。

后序遍历代码

二叉查找树(一)之 图文解析 和 C语言的实现
void postorder_bstree(BSTree tree)
{
    if(tree != NULL)
    {
        postorder_bstree(tree->left);
        postorder_bstree(tree->right);
        printf("%d ", tree->key);
    }
}
二叉查找树(一)之 图文解析 和 C语言的实现

 

 

下面通过例子对这些遍历方式进行介绍。

二叉查找树(一)之 图文解析 和 C语言的实现

对于上面的二叉树而言,
(01) 前序遍历结果: 3 1 2 5 4 6
(02) 中序遍历结果: 1 2 3 4 5 6 
(03) 后序遍历结果: 2 1 4 6 5 3

 

3. 查找

递归版本的代码

二叉查找树(一)之 图文解析 和 C语言的实现
Node* bstree_search(BSTree x, Type key)
{
    if (x==NULL || x->key==key)
        return x;

    if (key < x->key)
        return bstree_search(x->left, key);
    else
        return bstree_search(x->right, key);
}
二叉查找树(一)之 图文解析 和 C语言的实现

非递归版本的代码

二叉查找树(一)之 图文解析 和 C语言的实现
Node* iterative_bstree_search(BSTree x, Type key)
{
    while ((x!=NULL) && (x->key!=key))
    {
        if (key < x->key)
            x = x->left;
        else
            x = x->right;
    }

    return x;
}
二叉查找树(一)之 图文解析 和 C语言的实现

 

4. 最大值和最小值

查找最大值的代码

二叉查找树(一)之 图文解析 和 C语言的实现
Node* bstree_maximum(BSTree tree)
{
    if (tree == NULL)
        return NULL;

    while(tree->right != NULL)
        tree = tree->right;
    return tree;
}
二叉查找树(一)之 图文解析 和 C语言的实现

查找最小值的代码

二叉查找树(一)之 图文解析 和 C语言的实现
Node* bstree_minimum(BSTree tree)
{
    if (tree == NULL)
        return NULL;

    while(tree->left != NULL)
        tree = tree->left;
    return tree;
}
二叉查找树(一)之 图文解析 和 C语言的实现


5. 前驱和后继

节点的前驱:是该节点的左子树中的最大节点。
节点的后继:是该节点的右子树中的最小节点。

查找前驱节点的代码

二叉查找树(一)之 图文解析 和 C语言的实现
Node* bstree_predecessor(Node *x)
{
    // 如果x存在左孩子,则"x的前驱结点"为 "以其左孩子为根的子树的最大结点"。
    if (x->left != NULL)
        return bstree_maximum(x->left);

    // 如果x没有左孩子。则x有以下两种可能:
    // (01) x是"一个右孩子",则"x的前驱结点"为 "它的父结点"。
    // (01) x是"一个左孩子",则查找"x的最低的父结点,并且该父结点要具有右孩子",找到的这个"最低的父结点"就是"x的前驱结点"。
    Node* y = x->parent;
    while ((y!=NULL) && (x==y->left))
    {
        x = y;
        y = y->parent;
    }

    return y;
}
二叉查找树(一)之 图文解析 和 C语言的实现

查找后继节点的代码

二叉查找树(一)之 图文解析 和 C语言的实现
Node* bstree_successor(Node *x)
{
    // 如果x存在右孩子,则"x的后继结点"为 "以其右孩子为根的子树的最小结点"。
    if (x->right != NULL)
        return bstree_minimum(x->right);

    // 如果x没有右孩子。则x有以下两种可能:
    // (01) x是"一个左孩子",则"x的后继结点"为 "它的父结点"。
    // (02) x是"一个右孩子",则查找"x的最低的父结点,并且该父结点要具有左孩子",找到的这个"最低的父结点"就是"x的后继结点"。
    Node* y = x->parent;
    while ((y!=NULL) && (x==y->right))
    {
        x = y;
        y = y->parent;
    }

    return y;
}
二叉查找树(一)之 图文解析 和 C语言的实现

 

6. 插入

插入节点的代码

二叉查找树(一)之 图文解析 和 C语言的实现
static Node* bstree_insert(BSTree tree, Node *z)
{
    Node *y = NULL;
    Node *x = tree;

    // 查找z的插入位置
    while (x != NULL)
    {
        y = x;
        if (z->key < x->key)
            x = x->left;
        else
            x = x->right;
    }

    z->parent = y;
    if (y==NULL)
        tree = z;
    else if (z->key < y->key)
        y->left = z;
    else
        y->right = z;

    return tree;
}

Node* insert_bstree(BSTree tree, Type key)
{
    Node *z;    // 新建结点

    // 如果新建结点失败,则返回。
    if ((z=create_bstree_node(key, NULL, NULL, NULL)) == NULL)
        return tree;

    return bstree_insert(tree, z);
}
二叉查找树(一)之 图文解析 和 C语言的实现

bstree_insert(tree, z)是内部函数,它的作用是:将结点(z)插入到二叉树(tree)中,并返回插入节点后的根节点。
insert_bstree(tree, key)是对外接口,它的作用是:在树中新增节点,key是节点的值;并返回插入节点后的根节点。

 

注:本文实现的二叉查找树是允许插入相同键值的节点的!若用户不希望插入相同键值的节点,将bstree_insert()修改为以下代码即可。

二叉查找树(一)之 图文解析 和 C语言的实现
static Node* bstree_insert(BSTree tree, Node *z)
{
    Node *y = NULL;
    Node *x = tree;

    // 查找z的插入位置
    while (x != NULL)
    {
        y = x;
        if (z->key < x->key)
            x = x->left;
        else  if (z->key > x->key)
            x = x->right;
        else
        {
            free(z); // 释放之前分配的系统。
            return tree;
        }
    }

    z->parent = y;
    if (y==NULL)
        tree = z;
    else if (z->key < y->key)
        y->left = z;
    else
        y->right = z;

    return tree;
}
二叉查找树(一)之 图文解析 和 C语言的实现

 

7. 删除

删除节点的代码

二叉查找树(一)之 图文解析 和 C语言的实现
static Node* bstree_delete(BSTree tree, Node *z)
{
    Node *x=NULL;
    Node *y=NULL;

    if ((z->left == NULL) || (z->right == NULL) )
        y = z;
    else
        y = bstree_successor(z);

    if (y->left != NULL)
        x = y->left;
    else
        x = y->right;

    if (x != NULL)
        x->parent = y->parent;

    if (y->parent == NULL)
        tree = x;
    else if (y == y->parent->left)
        y->parent->left = x;
    else
        y->parent->right = x;

    if (y != z) 
        z->key = y->key;

    if (y!=NULL)
        free(y);

    return tree;
}

Node* delete_bstree(BSTree tree, Type key)
{
    Node *z, *node; 

    if ((z = bstree_search(tree, key)) != NULL)
        tree = bstree_delete(tree, z);

    return tree;
}
二叉查找树(一)之 图文解析 和 C语言的实现

bstree_delete(tree, z)是内部函数,它的作用是:删除二叉树(tree)中的节点(z),并返回删除节点后的根节点。
delete_bstree(tree, key)是对外接口,它的作用是:在树中查找键值为key的节点,找到的话就删除该节点;并返回删除节点后的根节点。


8. 打印

打印二叉树的代码

二叉查找树(一)之 图文解析 和 C语言的实现
void print_bstree(BSTree tree, Type key, int direction)
{
    if(tree != NULL)
    {
        if(direction==0)    // tree是根节点
            printf("%2d is root\n", tree->key);
        else                // tree是分支节点
            printf("%2d is %2d‘s %6s child\n", tree->key, key, direction==1?"right" : "left");

        print_bstree(tree->left, tree->key, -1);
        print_bstree(tree->right,tree->key,  1);
    }
}
二叉查找树(一)之 图文解析 和 C语言的实现

print_bstree(tree, key, direction)的作用是打印整颗二叉树(tree)。其中,tree是二叉树节点,key是二叉树的键值,而direction表示该节点的类型:

direction为 0,表示该节点是根节点;
direction为-1,表示该节点是它的父结点的左孩子;
direction为 1,表示该节点是它的父结点的右孩子。

 

9. 销毁二叉树

销毁二叉树的代码

二叉查找树(一)之 图文解析 和 C语言的实现
void destroy_bstree(BSTree tree)
{
    if (tree==NULL)
        return ;

    if (tree->left != NULL)
        destroy_bstree(tree->left);
    if (tree->right != NULL)
        destroy_bstree(tree->right);

    free(tree);
}
二叉查找树(一)之 图文解析 和 C语言的实现

 

完整的实现代码

二叉查找树的头文件(bstree.h)

二叉查找树(一)之 图文解析 和 C语言的实现
 1 #ifndef _BINARY_SEARCH_TREE_H_
 2 #define _BINARY_SEARCH_TREE_H_
 3 
 4 typedef int Type;
 5 
 6 typedef struct BSTreeNode{
 7     Type   key;                    // 关键字(键值)
 8     struct BSTreeNode *left;    // 左孩子
 9     struct BSTreeNode *right;    // 右孩子
10     struct BSTreeNode *parent;    // 父结点
11 }Node, *BSTree;
12 
13 // 前序遍历"二叉树"
14 void preorder_bstree(BSTree tree);
15 // 中序遍历"二叉树"
16 void inorder_bstree(BSTree tree);
17 // 后序遍历"二叉树"
18 void postorder_bstree(BSTree tree);
19 
20 // (递归实现)查找"二叉树x"中键值为key的节点
21 Node* bstree_search(BSTree x, Type key);
22 // (非递归实现)查找"二叉树x"中键值为key的节点
23 Node* iterative_bstree_search(BSTree x, Type key);
24 
25 // 查找最小结点:返回tree为根结点的二叉树的最小结点。
26 Node* bstree_minimum(BSTree tree);
27 // 查找最大结点:返回tree为根结点的二叉树的最大结点。
28 Node* bstree_maximum(BSTree tree);
29 
30 // 找结点(x)的后继结点。即,查找"二叉树中数据值大于该结点"的"最小结点"。
31 Node* bstree_successor(Node *x);
32 // 找结点(x)的前驱结点。即,查找"二叉树中数据值小于该结点"的"最大结点"。
33 Node* bstree_predecessor(Node *x);
34 
35 // 将结点插入到二叉树中,并返回根节点
36 Node* insert_bstree(BSTree tree, Type key);
37 
38 // 删除结点(key为节点的值),并返回根节点
39 Node* delete_bstree(BSTree tree, Type key);
40 
41 // 销毁二叉树
42 void destroy_bstree(BSTree tree);
43 
44 // 打印二叉树
45 void print_bstree(BSTree tree, Type key, int direction);
46 
47 #endif
View Code

二叉查找树的实现文件(bstree.c)

二叉查找树(一)之 图文解析 和 C语言的实现
  1 /**
  2  * 二叉搜索树(C语言): C语言实现的二叉搜索树。
  3  *
  4  * @author skywang
  5  * @date 2013/11/07
  6  */
  7 
  8 #include <stdio.h>
  9 #include <stdlib.h>
 10 #include "bstree.h"
 11 
 12 
 13 /*
 14  * 前序遍历"二叉树"
 15  */
 16 void preorder_bstree(BSTree tree)
 17 {
 18     if(tree != NULL)
 19     {
 20         printf("%d ", tree->key);
 21         preorder_bstree(tree->left);
 22         preorder_bstree(tree->right);
 23     }
 24 }
 25 
 26 /*
 27  * 中序遍历"二叉树"
 28  */
 29 void inorder_bstree(BSTree tree)
 30 {
 31     if(tree != NULL)
 32     {
 33         inorder_bstree(tree->left);
 34         printf("%d ", tree->key);
 35         inorder_bstree(tree->right);
 36     }
 37 }
 38 
 39 /*
 40  * 后序遍历"二叉树"
 41  */
 42 void postorder_bstree(BSTree tree)
 43 {
 44     if(tree != NULL)
 45     {
 46         postorder_bstree(tree->left);
 47         postorder_bstree(tree->right);
 48         printf("%d ", tree->key);
 49     }
 50 }
 51 
 52 /*
 53  * (递归实现)查找"二叉树x"中键值为key的节点
 54  */
 55 Node* bstree_search(BSTree x, Type key)
 56 {
 57     if (x==NULL || x->key==key)
 58         return x;
 59 
 60     if (key < x->key)
 61         return bstree_search(x->left, key);
 62     else
 63         return bstree_search(x->right, key);
 64 }
 65 
 66 /*
 67  * (非递归实现)查找"二叉树x"中键值为key的节点
 68  */
 69 Node* iterative_bstree_search(BSTree x, Type key)
 70 {
 71     while ((x!=NULL) && (x->key!=key))
 72     {
 73         if (key < x->key)
 74             x = x->left;
 75         else
 76             x = x->right;
 77     }
 78 
 79     return x;
 80 }
 81 
 82 /* 
 83  * 查找最小结点:返回tree为根结点的二叉树的最小结点。
 84  */
 85 Node* bstree_minimum(BSTree tree)
 86 {
 87     if (tree == NULL)
 88         return NULL;
 89 
 90     while(tree->left != NULL)
 91         tree = tree->left;
 92     return tree;
 93 }
 94  
 95 /* 
 96  * 查找最大结点:返回tree为根结点的二叉树的最大结点。
 97  */
 98 Node* bstree_maximum(BSTree tree)
 99 {
100     if (tree == NULL)
101         return NULL;
102 
103     while(tree->right != NULL)
104         tree = tree->right;
105     return tree;
106 }
107 
108 /* 
109  * 找结点(x)的后继结点。即,查找"二叉树中数据值大于该结点"的"最小结点"。
110  */
111 Node* bstree_successor(Node *x)
112 {
113     // 如果x存在右孩子,则"x的后继结点"为 "以其右孩子为根的子树的最小结点"。
114     if (x->right != NULL)
115         return bstree_minimum(x->right);
116 
117     // 如果x没有右孩子。则x有以下两种可能:
118     // (01) x是"一个左孩子",则"x的后继结点"为 "它的父结点"。
119     // (02) x是"一个右孩子",则查找"x的最低的父结点,并且该父结点要具有左孩子",找到的这个"最低的父结点"就是"x的后继结点"。
120     Node* y = x->parent;
121     while ((y!=NULL) && (x==y->right))
122     {
123         x = y;
124         y = y->parent;
125     }
126 
127     return y;
128 }
129  
130 /* 
131  * 找结点(x)的前驱结点。即,查找"二叉树中数据值小于该结点"的"最大结点"。
132  */
133 Node* bstree_predecessor(Node *x)
134 {
135     // 如果x存在左孩子,则"x的前驱结点"为 "以其左孩子为根的子树的最大结点"。
136     if (x->left != NULL)
137         return bstree_maximum(x->left);
138 
139     // 如果x没有左孩子。则x有以下两种可能:
140     // (01) x是"一个右孩子",则"x的前驱结点"为 "它的父结点"。
141     // (01) x是"一个左孩子",则查找"x的最低的父结点,并且该父结点要具有右孩子",找到的这个"最低的父结点"就是"x的前驱结点"。
142     Node* y = x->parent;
143     while ((y!=NULL) && (x==y->left))
144     {
145         x = y;
146         y = y->parent;
147     }
148 
149     return y;
150 }
151 
152 /*
153  * 创建并返回二叉树结点。
154  *
155  * 参数说明:
156  *     key 是键值。
157  *     parent 是父结点。
158  *     left 是左孩子。
159  *     right 是右孩子。
160  */
161 static Node* create_bstree_node(Type key, Node *parent, Node *left, Node* right)
162 {
163     Node* p;
164 
165     if ((p = (Node *)malloc(sizeof(Node))) == NULL)
166         return NULL;
167     p->key = key;
168     p->left = left;
169     p->right = right;
170     p->parent = parent;
171 
172     return p;
173 }
174 
175 /* 
176  * 将结点插入到二叉树中
177  *
178  * 参数说明:
179  *     tree 二叉树的根结点
180  *     z 插入的结点
181  * 返回值:
182  *     根节点
183  */
184 static Node* bstree_insert(BSTree tree, Node *z)
185 {
186     Node *y = NULL;
187     Node *x = tree;
188 
189     // 查找z的插入位置
190     while (x != NULL)
191     {
192         y = x;
193         if (z->key < x->key)
194             x = x->left;
195         else
196             x = x->right;
197     }
198 
199     z->parent = y;
200     if (y==NULL)
201         tree = z;
202     else if (z->key < y->key)
203         y->left = z;
204     else
205         y->right = z;
206 
207     return tree;
208 }
209 
210 /* 
211  * 新建结点(key),并将其插入到二叉树中
212  *
213  * 参数说明:
214  *     tree 二叉树的根结点
215  *     key 插入结点的键值
216  * 返回值:
217  *     根节点
218  */
219 Node* insert_bstree(BSTree tree, Type key)
220 {
221     Node *z;    // 新建结点
222 
223     // 如果新建结点失败,则返回。
224     if ((z=create_bstree_node(key, NULL, NULL, NULL)) == NULL)
225         return tree;
226 
227     return bstree_insert(tree, z);
228 }
229 
230 /* 
231  * 删除结点(z),并返回根节点
232  *
233  * 参数说明:
234  *     tree 二叉树的根结点
235  *     z 删除的结点
236  * 返回值:
237  *     根节点
238  */
239 static Node* bstree_delete(BSTree tree, Node *z)
240 {
241     Node *x=NULL;
242     Node *y=NULL;
243 
244     if ((z->left == NULL) || (z->right == NULL) )
245         y = z;
246     else
247         y = bstree_successor(z);
248 
249     if (y->left != NULL)
250         x = y->left;
251     else
252         x = y->right;
253 
254     if (x != NULL)
255         x->parent = y->parent;
256 
257     if (y->parent == NULL)
258         tree = x;
259     else if (y == y->parent->left)
260         y->parent->left = x;
261     else
262         y->parent->right = x;
263 
264     if (y != z) 
265         z->key = y->key;
266 
267     if (y!=NULL)
268         free(y);
269 
270     return tree;
271 }
272 
273 /* 
274  * 删除结点(key为节点的键值),并返回根节点
275  *
276  * 参数说明:
277  *     tree 二叉树的根结点
278  *     z 删除的结点
279  * 返回值:
280  *     根节点
281  */
282 Node* delete_bstree(BSTree tree, Type key)
283 {
284     Node *z, *node; 
285 
286     if ((z = bstree_search(tree, key)) != NULL)
287         tree = bstree_delete(tree, z);
288 
289     return tree;
290 }
291 
292 /*
293  * 销毁二叉树
294  */
295 void destroy_bstree(BSTree tree)
296 {
297     if (tree==NULL)
298         return ;
299 
300     if (tree->left != NULL)
301         destroy_bstree(tree->left);
302     if (tree->right != NULL)
303         destroy_bstree(tree->right);
304 
305     free(tree);
306 }
307 
308 /*
309  * 打印"二叉树"
310  *
311  * tree       -- 二叉树的节点
312  * key        -- 节点的键值 
313  * direction  --  0,表示该节点是根节点;
314  *               -1,表示该节点是它的父结点的左孩子;
315  *                1,表示该节点是它的父结点的右孩子。
316  */
317 void print_bstree(BSTree tree, Type key, int direction)
318 {
319     if(tree != NULL)
320     {
321         if(direction==0)    // tree是根节点
322             printf("%2d is root\n", tree->key);
323         else                // tree是分支节点
324             printf("%2d is %2d‘s %6s child\n", tree->key, key, direction==1?"right" : "left");
325 
326         print_bstree(tree->left, tree->key, -1);
327         print_bstree(tree->right,tree->key,  1);
328     }
329 }
View Code

二叉查找树的测试程序(btree_test.c)

二叉查找树(一)之 图文解析 和 C语言的实现
 1 /**
 2  * C 语言: 二叉查找树
 3  *
 4  * @author skywang
 5  * @date 2013/11/07
 6  */
 7 
 8 #include <stdio.h>
 9 #include "bstree.h"
10 
11 static int arr[]= {1,5,4,3,2,6};
12 #define TBL_SIZE(a) ( (sizeof(a)) / (sizeof(a[0])) )
13 
14 void main()
15 {
16     int i, ilen;
17     BSTree root=NULL;
18 
19     printf("== 依次添加: ");
20     ilen = TBL_SIZE(arr);
21     for(i=0; i<ilen; i++)
22     {
23         printf("%d ", arr[i]);
24         root = insert_bstree(root, arr[i]);
25     }
26 
27     printf("\n== 前序遍历: ");
28     preorder_bstree(root);
29 
30     printf("\n== 中序遍历: ");
31     inorder_bstree(root);
32 
33     printf("\n== 后序遍历: ");
34     postorder_bstree(root);
35     printf("\n");
36 
37     printf("== 最小值: %d\n", bstree_minimum(root)->key);
38     printf("== 最大值: %d\n", bstree_maximum(root)->key);
39     printf("== 树的详细信息: \n");
40     print_bstree(root, root->key, 0);
41 
42     printf("\n== 删除根节点: %d", arr[3]);
43     root = delete_bstree(root, arr[3]);
44 
45     printf("\n== 中序遍历: ");
46     inorder_bstree(root);
47     printf("\n");
48 
49     // 销毁二叉树
50     destroy_bstree(root);
51 }
View Code

 

二叉查找树的C测试程序

上面的btree_test.c是二叉查找树的测试程序,它的运行结果如下:

二叉查找树(一)之 图文解析 和 C语言的实现
== 依次添加: 1 5 4 3 2 6 
== 前序遍历: 1 5 4 3 2 6 
== 中序遍历: 1 2 3 4 5 6 
== 后序遍历: 2 3 4 6 5 1 
== 最小值: 1
== 最大值: 6
== 树的详细信息: 
 1 is root
 5 is  1s  right child
 4 is  5s   left child
 3 is  4s   left child
 2 is  3s   left child
 6 is  5s  right child

== 删除根节点: 3
== 中序遍历: 1 2 4 5 6 
二叉查找树(一)之 图文解析 和 C语言的实现

 

下面对测试程序的流程进行分析!

(01) 新建"二叉查找树"root。

 

(02) 向二叉查找树中依次插入1,5,4,3,2,6 。如下图所示:

二叉查找树(一)之 图文解析 和 C语言的实现

 

(03) 打印树的信息
插入1,5,4,3,2,6之后,得到的二叉查找树如下:

二叉查找树(一)之 图文解析 和 C语言的实现

前序遍历结果: 1 5 4 3 2 6
中序遍历结果: 1 2 3 4 5 6
后序遍历结果: 2 3 4 6 5 1
最小值是1,而最大值是6。

 

 

(04) 删除节点3。如下图所示:

二叉查找树(一)之 图文解析 和 C语言的实现

 

(05) 重新遍历该二叉查找树。
中序遍历结果: 1 2 4 5 6

二叉查找树(一)之 图文解析 和 C语言的实现,布布扣,bubuko.com

二叉查找树(一)之 图文解析 和 C语言的实现

上一篇:智能优化算法:饥饿游戏搜索算法-附代码


下一篇:org.apache.shiro.SecurityUtils.getSubject().getSession()