一、二叉搜索树的定义及性质

二叉查找树(Binary Search Tree),也称有序二叉树(ordered binary tree),排序二叉树(sorted binary tree),是指一棵空树或者具有下列性质的二叉树:

1. 每个节点都有一个作为搜索依据的关键码( key) , 所有节点的关键码互不相同。
2. 左子树上所有节点的关键码( key) 都小于根节点的关键码( key) 。
3. 右子树上所有节点的关键码( key) 都大于根节点的关键码( key) 。
4. 左右子树都是二叉搜索树。

在实现中,由它的性质可以初步简单的定义出节点信息,我们需要定义一个内部类BinarySearchTreeNode,它包含两个分别指向左右节点的Node->_left和Node->_right,一个保存键值信息的Key。

 template <class K>
struct BinarySearchTreeNode
{
BinarySearchTreeNode<K>* _left; //指向左子树
BinarySearchTreeNode<K>* _right; //指向右子树
K _key; //关键码
BinarySearchTreeNode(const K& key)
:_left(NULL)
,_right(NULL)
,_key(key)
{}
};

如下图,这个是普通的二叉树,它具有普通二叉树的所有性质。

在此基础上,加上上面所述节点之间的大小关系,就是二叉查找树:

下面这个是二叉搜索树,可以很清晰的看出节点之间的大小关系:

而二叉搜索树里面真正有意义的是对其节点的增、删、查找等操作,下面我分别来介绍这几种算法原理。

二、算法操作

由内部节点构建二叉搜索树如下:

 template<class K>
class BinarySearchTree
{
typedef BinarySearchTreeNode<K> Node;
public:
BinarySearchTree()
:_root(NULL)
{}
~BinarySearchTree()
{
_Delete(_root);
}
//非递归
bool Insert(const K& key) //插入
{
return _Insert(_root, key);
}
bool Remove(const K& key) //删除
{
return _Remove(_root, key);
}
const Node* Find(const K& key) //查找
{
return _Find(_root, key);
}
//递归
bool InsertR(const K& key)
{
return _InsertR(_root, key);
}
bool RemoveR(const K& key)
{
return _RemoveR(_root, key);
}
const Node* FindR(const K& key)
{
return _FindR(_root, key);
}
void Inorder() //遍历打印
{
_Inorder(_root);
cout << endl;
}
private:
void _Delete(Node*& root);
bool _Insert(Node* root, const K& key);
bool _Remove(Node* root, const K& key);
Node* _Find(Node* root, const K& key);
bool _InsertR(Node*& root, const K& key);
bool _RemoveR(Node* & root, const K& key);
const Node*_FindR(Node* root, const K& key);
void _Inorder(Node* root);
private:
Node* _root;
};

1.查找

查找操作和二分查找类似,从根节点开始,将root->_key和要找的节点的key比较,如果root->_key小于要找的节点的key,那么就在左子树 _root->_left节点中查找,如果root->_key大于要找的节点的key,则在右子树_root->_right节点中查找,如果_root->_key和要找的节点的key相等,直接返回此节点。

查找操作图示如下:

该方法实现有递归和非递归两种。

非递归算法程序如下:

          const Node* Find(const K& key)
{
return _Find(_root, key);
}
Node* _Find(Node* root, const K& key)
{
while (root) {
if (root->_key > key) //向左查找
root = root->_left;
if (root->_key < key) //向右查找
root = root->_right;
else
return root; //找到节点
}
return NULL;
}

递归算法:

         const Node* FindR(const K& key)
{
return _FindR(_root, key);
}
const Node*_FindR(Node* root, const K& key)
{
if (root == NULL)
return NULL;
if (root->_key > key)
return _FindR(root->_left, key); //向左递归查找
if (root->_key < key)
return _FindR(root->_right, key); //向右递归查找
else
return root; //找到该节点
}

2.插入

插入一个节点第一步与与查找类似,首先要找到应该插入节点的位置,因为插入节点后不能破坏二叉搜索树的性质。首先判断如果树为空,则直接动态开辟节点并初始化为key插入即可,插入的节点即为根节点。然后再向下查找应该插入节点的位置,查找方法与上一个查找算法类似,不同的是在查找的过程中要将应该插入的位置的父节点记录下来。找到该位置后,将parent->_key与插入的key进行比较,判断应该插入到父节点的左子树还是右子树。如果parent->_key小于要插入节点的key,那么就插入为父节点的右子树parent->_right = new Node(key);  //插入右子树;如果parent->_key大于要插入节点的key,那么就插入为父节点的左子树parent->_left = new Node(key);   //插入左子树。

插入操作图示如下:

同样,插入操作的实现也有递归和非递归两种方法。

非递归法实现如下:

          bool Insert(const K& key)
{
return _Insert(_root, key);
}
bool Insert(Node*& root, const K& key) {
if (root == NULL){ //当树为空时,直接插入
root = new Node(key);
return true;
}
Node* cur = root;
Node* parent = cur;
while (cur) { //找到要插入的位置
if (cur->_key > key) {
parent = cur;
cur = cur->_left;
}
else if (cur->_key < key) {
parent = cur;
cur = cur->_right;
}
else
return false;
}
if (parent->_key < key) //找到插入位置的父节点,判断应该是父节点的左子树还是右子树右
parent->_right = new Node(key); //插入右子树
else
parent->_left = new Node(key); //插入左子树
return true;
}

递归法实现如下:

         bool InsertR(const K& key)
{
return _InsertR(_root, key);
}
bool _InsertR(Node*& root, const K& key) //注意:这里参数传递方法是传引用
{
if (root == NULL){
root = new Node(key);
return true;
}
if (root->_key > key)
return _InsertR(root->_left, key);
if (root->_key < key)
return _InsertR(root->_right, key);
else
return false;
}

3.删除

删除元素操作在二叉树的操作中应该是比较复杂的。但是只要一步一步分析的话其实也是比较容易实现的。首先判断该树是否为空,为空的话直接返回删除失败;该是不为空时,第一步是找到需要删除的节点,方法与前面的查找算法类似,不同的是也需要将删除的节点的父节点记录下来,以判断该节点删除以后需要调整父节点的哪一个子树。找到该节点后,复杂的地方才刚开始。

(1)当要删除的节点的左子树为空时,判断是否为该节点是否为根节点,若是,则直接删除,其右子树的根节点成为新的根节点;若不是则再次判断该节点在其父节点的哪个子树上。若该节点在其父节点的左子树上,则将该节点的右子树连到该节点的父节点的左子树上,然后将该节点删除;若该节点在其父节点的右子树上,则将该节点的右子树连到该节点的父节点的右子树上,然后将该节点删除。

(2)当要删除的节点的右子树为空时,判断是否为该节点是否为根节点,若是,则直接删除,其左子树的根节点成为新的根节点;若不是则再次判断该节点在其父节点的哪个子树上。若该节点在其父节点的左子树上,则将该节点的左子树连到该节点的父节点的左子树上,然后将该节点删除;若该节点在其父节点的右子树上,则将该节点的左子树连到该节点的父节点的右子树上,然后将该节点删除。(类似于(1))。

(3)当要删除的节点的左右子树都不为为空时,首先查找该节点的右子树的最小节点,即找该节点的右子树的最左节点,让这个最小节点代替该节点的位置,然后将此最小节点删除,这样调整之后才能使此树依然为搜索二叉树,另外在查找最左节点时应将它的父节点记录下来(原理同(1)和(2))。找到该节点的右子树的最左节点后,将最左节点的值赋给要删除的节点,作为新的该节点,其原值被覆盖,后面再删除的就是找到的最左节点。删除之前判断若此最左节点在其父节点的左子树上,则将该节点的右子树连到该节点的父节点的左子树上,然后将该节点删除;若该节点在其父节点的右子树上(即右子树最左节点为右子树的根节点),则将该节点的右子树连到该节点的父节点的右子树上,然后将该节点删除。(还有另一种方法,就是找到该节点的右子树的最大节点,即最右节点,与该节点替换后再删除,原理相同,此处不在赘述)。

此原理叙述起来较复杂,下面请看图示情况分类:

当删除的节点只有1个子节点时,在左边和在右边原理类似:

当删除的节点有2个子节点时:

用具体的二叉查找树举例如下:

非递归法代码如下:

                 bool Remove(const K& key) {
if (_root == NULL)
return false;
Node* del = _root;
Node* parent = del;
while (del) {
if (del->_key < key) { //向右搜索
parent = del;
del = del->_right;
}
if (del->_key > key) { //向左搜索
parent = del;
del = del->_left;
}
if (del->_key == key) { //要删除的节点找到
Node* cur = del;
if (cur->_left == NULL) { //当此节点左子树为空
if (_root->_key == key) //删除根节点
_root = _root->_right;
else{
if(parent->_left == cur)
parent->_left = cur->_right; //当找到的节点在其父节点的左子树上
else
parent->_right = cur->_right; //当找到的节点在其父节点的右子树上
}
}
else if (cur->_right == NULL) { //当此节点右子树为空
if (_root->_key == key)
_root = _root->_left;
else {
if (parent->_left == cur)
parent->_left = cur->_left;
else
parent->_right = cur->_left;
}
}
else{ //左右子树都不为空
cur = cur->_right;
while (cur->_left) { //找右子树的最左节点
parent = cur;
cur = cur->_left;
}
del->_key = cur->_key; //将最左节点的值给要删除的节点,作为新的该节点,其原值被覆盖
del = cur; //后面删除的就是找到的最左节点
if (parent->_left == cur) //此时的cur是最左节点
parent->_left = cur->_right;
else
parent->_right = cur->_right; //右子树最左节点为右子树的根节点
}
delete del;
del = NULL;
return true;
}
}
return false;
}

递归法删除节点代码如下:

         bool RemoveR(const K& key)
{
return _RemoveR(_root, key);
}
bool _RemoveR(Node* & root, const K& key) //注意此处传引用
{
if (root == NULL)
return false;
if (root->_key > key)
return _RemoveR(root->_left, key); //向左递归搜索
if (root->_key < key)
return _RemoveR(root->_right, key); //向右递归搜索
else{ //要删除的节点找到
Node* del = root;
if (root->_left == NULL)
root = root->_right;
if (root->_right == NULL)
root = root->_left;
else{
Node* cur = root;
Node* parent = cur;
cur = cur->_right;
while (cur->_left) {
parent = cur;
cur = cur->_left;
}
del->_key = cur->_key;
del = cur;
if (parent->_left = cur)
parent->_left = cur->_right;
else
parent->_right = cur->_right;
}
delete del;
del = NULL;
return true;
}
}

三、算法分析与总结

二叉查找树的运行时间和树的形状有关,树的形状又和插入元素的顺序有关。在最好的情况下,节点完全平衡,从根节点到最底层叶子节点只有lgN个节点。在最差的情况下,根节点到最底层叶子节点会有N各节点。在一般情况下,树的形状和最好的情况接近。

在分析二叉查找树的时候,我们通常会假设插入的元素顺序是随机的。对于N个不同元素,随机插入的二叉查找树来说,其平均查找/插入的时间复杂度大约为2lnN。

前面有对二分查找时间复杂度的分析,对二叉查找树的理解可以类比于此。它和二分查找一样,插入和查找的时间复杂度均为lgN,但是在最坏的情况下仍然会有N的时间复杂度。原因在于插入和删除元素的时候,树没有保持平衡。我们追求的是在最坏的情况下仍然有较好的时间复杂度,这就是后面要讲的平衡查找树的内容了。

下面是二叉查找树的时间复杂度:

四、完整的二叉搜索树代码

 #include<iostream>
using namespace std; template <class K>
struct BinarySearchTreeNode
{
BinarySearchTreeNode<K>* _left;
BinarySearchTreeNode<K>* _right;
K _key;
BinarySearchTreeNode(const K& key)
:_left(NULL)
,_right(NULL)
,_key(key)
{}
}; template<class K>
class BinarySearchTree
{
typedef BinarySearchTreeNode<K> Node;
public:
BinarySearchTree()
:_root(NULL)
{}
~BinarySearchTree()
{
_Delete(_root);
}
//非递归
bool Insert(const K& key)
{
return _Insert(_root, key);
}
bool Remove(const K& key)
{
return _Remove(_root, key);
}
const Node* Find(const K& key)
{
return _Find(_root, key);
}
//递归
bool InsertR(const K& key)
{
return _InsertR(_root, key);
}
bool RemoveR(const K& key)
{
return _RemoveR(_root, key);
}
const Node* FindR(const K& key)
{
return _FindR(_root, key);
}
void Inorder()
{
_Inorder(_root);
cout << endl;
}
private:
void _Delete(Node*& root)
{
if (root)
{
_Delete(root->_left);
_Delete(root->_right);
delete root;
root = NULL;
}
return;
}
bool Insert(const K& key) {
if (_root == NULL){ //当树为空时,直接插入
_root = new Node(key);
return true;
}
Node* cur = _root;
Node* parent = cur;
while (cur) { //找到要插入的位置
if (cur->_key > key) {
parent = cur;
cur = cur->_left;
}
else if (cur->_key < key) {
parent = cur;
cur = cur->_right;
}
else
return false;
}
if (parent->_key < key) //找到插入位置的父节点,判断应该是父节点的左子树还是右子树右
parent->_right = new Node(key); //插入右子树
else
parent->_left = new Node(key); //插入左子树
return true;
} bool Remove(const K& key) {
if (_root == NULL)
return false;
Node* del = _root;
Node* parent = del;
while (del) {
if (del->_key < key) { //向右搜索
parent = del;
del = del->_right;
}
if (del->_key > key) { //向左搜索
parent = del;
del = del->_left;
}
if (del->_key == key) { //要删除的节点找到
Node* cur = del;
if (cur->_left == NULL) { //当此节点左子树为空
if (_root->_key == key) //删除根节点
_root = _root->_right;
else{
if(parent->_left == cur)
parent->_left = cur->_right; //当找到的节点在其父节点的左子树上
else
parent->_right = cur->_right; //当找到的节点在其父节点的右子树上
}
}
else if (cur->_right == NULL) { //当此节点右子树为空
if (_root->_key == key)
_root = _root->_left;
else {
if (parent->_left == cur)
parent->_left = cur->_left;
else
parent->_right = cur->_left;
}
}
else{ //左右子树都不为空
cur = cur->_right;
while (cur->_left) { //找右子树的最左节点
parent = cur;
cur = cur->_left;
}
del->_key = cur->_key; //将最左节点的值给要删除的节点,作为新的该节点,其原值被覆盖
del = cur; //后面删除的就是找到的最左节点
if (parent->_left == cur) //此时的cur是最左节点
parent->_left = cur->_right;
else
parent->_right = cur->_right; //右子树最左节点为右子树的根节点
}
delete del;
del = NULL;
return true;
}
}
return false;
}
Node* _Find(Node* root, const K& key)
{
while (root) {
if (root->_key > key) //向左查找
root = root->_left;
if (root->_key < key) //向右查找
root = root->_right;
else
return root; //找到节点
}
return NULL;
} bool _InsertR(Node*& root, const K& key)
{
if (root == NULL){
root = new Node(key);
return true;
}
if (root->_key > key)
return _InsertR(root->_left, key);
if (root->_key < key)
return _InsertR(root->_right, key);
else
return false;
}
bool _RemoveR(Node* & root, const K& key)
{
if (root == NULL)
return false;
if (root->_key > key)
return _RemoveR(root->_left, key); //向左递归搜索
if (root->_key < key)
return _RemoveR(root->_right, key); //向右递归搜索
else{ //要删除的节点找到
Node* del = root;
if (root->_left == NULL)
root = root->_right;
if (root->_right == NULL)
root = root->_left;
else{
Node* cur = root;
Node* parent = cur;
cur = cur->_right;
while (cur->_left) {
parent = cur;
cur = cur->_left;
}
del->_key = cur->_key;
del = cur;
if (parent->_left = cur)
parent->_left = cur->_right;
else
parent->_right = cur->_right;
}
delete del;
del = NULL;
return true;
} }
const Node*_FindR(Node* root, const K& key)
{
if (root == NULL)
return NULL;
if (root->_key > key)
return _FindR(root->_left, key);
if (root->_key < key)
return _FindR(root->_right, key);
else
return root;
}
void _Inorder(Node* root)
{
if (root)
{
_Inorder(root->_left);
cout << root->_key << " ";
_Inorder(root->_right);
}
}
private:
Node* _root;
};
//测试代码:
#include"SearchTree.h" void Test()
{
BinarySearchTree<int> tree;
tree.InsertR();
tree.InsertR();
tree.Insert();
tree.InsertR();
tree.Insert();
tree.InsertR();
tree.Insert();
tree.InsertR();
tree.Insert(); cout << tree.FindR() << endl;
cout << tree.FindR() << endl;
cout << tree.FindR() << endl;
cout << tree.FindR() << endl;
cout << tree.Find() << endl;
cout << tree.Find() << endl;
cout << tree.Find() << endl;
cout << tree.Find() << endl; tree.Inorder(); tree.RemoveR();
tree.RemoveR();
tree.RemoveR();
tree.Remove();
tree.Remove();
tree.Remove();
tree.Remove();
tree.Remove();
tree.Remove(); tree.~BinarySearchTree(); }
int main()
{
Test();
getchar();
return ;
}

部分测试代码的输出结果:

本文部分图示来自于:http://www.cnblogs.com/yangecnu/p/Introduce-Binary-Search-Tree.htm

BinarySearchTree-二叉搜索树的更多相关文章

  1. BinarySearchTree二叉搜索树的实现

    /* 二叉搜索树(Binary Search Tree),(又:二叉查找树,二叉排序树)它或者是一棵空树,或者是具有下列性质的二叉树: 若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值; ...

  2. 二叉搜索树详解(Java实现)

    1.二叉搜索树定义 二叉搜索树,是指一棵空树或者具有下列性质的二叉树: 若任意节点的左子树不空,则左子树上所有节点的值均小于它的根节点的值: 若任意节点的右子树不空,则右子树上所有节点的值均大于它的根 ...

  3. BinarySearchTree(二叉搜索树)原理及C++代码实现

    BST是一类用途极广的数据结构.它有如下性质:设x是二叉搜索树内的一个结点.如果y是x左子树中的一个结点,那么y.key<=x.key.如果y是x右子树中的一个结点,那么y.key>=x. ...

  4. javascript数据结构——写一个二叉搜索树

    二叉搜索树就是左侧子节点值比根节点值小,右侧子节点值比根节点值大的二叉树. 照着书敲了一遍. function BinarySearchTree(){ var Node = function(key) ...

  5. Java二叉搜索树实现

    树集合了数组(查找速度快)和链表(插入.删除速度快)的优点 二叉树是一种特殊的树,即:树中的每个节点最多只能有两个子节点 二叉搜索树是一种特殊的二叉树,即:节点的左子节点的值都小于这个节点的值,节点的 ...

  6. 数据结构-二叉树(应用篇)-之二叉搜索树 C和C++的实现

    一.概念 二叉搜索树(Binary Sort Tree/Binary Search Tree...),是二叉树的一种特殊扩展.也是一种动态查找表. 在二叉搜索树中,左子树上所有节点的均小于根节点,右子 ...

  7. Java与算法之(13) - 二叉搜索树

    查找是指在一批记录中找出满足指定条件的某一记录的过程,例如在数组{ 8, 4, 12, 2, 6, 10, 14, 1, 3, 5, 7, 9, 11, 13, 15 }中查找数字15,实现代码很简单 ...

  8. 二叉搜索树(Java实现)

    二叉搜索树基本操作 求树中的结点个数 判断节点是否为空 向树中插入新结点key-value 树中是否存在key 返回树中key对应的value值 先序遍历 中序遍历 后续遍历 层序遍历 求树中key最 ...

  9. [数据结构]P2.1 二叉搜索树

    二叉树就是每个节点最多有两个分叉的树.这里我们写一写一个典型的例子二叉搜索树,它存在的实际意义是什么呢? 在P1.1链表中,我们清楚了链表的优势是善于删除添加节点,但是其取值很慢:数组的优势是善于取值 ...

  10. 【IT笔试面试题整理】二叉搜索树转换为双向链表

    [试题描述] 将二叉搜索树转换为双向链表 对于二叉搜索树,可以将其转换为双向链表,其中,节点的左子树指针在链表中指向前一个节点,右子树指针在链表中指向后一个节点. 思路一: 采用递归思想,对于二叉搜索 ...

随机推荐

  1. 用Pyton玩转数据练习题---第二周

    找前5个默尼森数.P是素数且M也是素数,并且满足等式M=2**P-1,则称M为默尼森数.例如,P=5,M=2**P-1=31,5和31都是素数,因此31是默尼森数. # coding:utf-8 im ...

  2. 清北Day 2

    清北第二天,感受到了来自这个世界的不友善,大概把没听过不会的"名词"记录下来就已经一面了,然后被大佬说这都是最基础的东西,就很皮,那就趁别人练习字符串的题的时候,来写波博客了,倒不 ...

  3. 权限管理系统 mysql 数据脚本

    # SQL-Front 5.1 (Build 4.16) /*!40101 SET @OLD_SQL_MODE=@@SQL_MODE */; /*!40101 SET SQL_MODE='STRICT ...

  4. 【lucene系列学习四】使用IKAnalyzer分词器实现敏感词和停用词过滤

    Lucene自带的中文分词器SmartChineseAnalyzer不太好扩展,于是我用了IKAnalyzer来进行敏感词和停用词的过滤. 首先,下载IKAnalyzer,我下载了 然后,由于IKAn ...

  5. Linux中批量添加文件和修改文件小笔记

    1.#使用通配符批量创建5个文件 $ touch file{1..10}.txt 2.#批量修改这五个后缀名为.txt的文本文件名重命名为.c后缀名结尾的文件 $ rename 's/\.txt/\. ...

  6. js原型链部分详细使用说明案例

    1. 'index.html'文件 ```html <!DOCTYPE html> <html lang="en"> <head> <me ...

  7. [刷题]算法竞赛入门经典 3-1/UVa1585 3-2/UVa1586 3-3/UVa1225

    书上具体所有题目:http://pan.baidu.com/s/1hssH0KO(我也是在网上找到的pdf,但不记得是从哪里搜刮到的了,就重新上传了一遍) PS:第一次写博客分享我的代码,不知道我对c ...

  8. 通过一步步创建sharded cluster来认识mongodb

    mongodb是目前使用非常广泛的nosql(not only sql)之一,在db engines上排名非常靠前,下图是5月份的排名: 可以看到前面四个都是传统的关系型数据库,而mongodb在no ...

  9. SparkMLlib分类算法之支持向量机

    SparkMLlib分类算法之支持向量机 (一),概念 支持向量机(support vector machine)是一种分类算法,通过寻求结构化风险最小来提高学习机泛化能力,实现经验风险和置信范围的最 ...

  10. 【AngularJS中的自定义服务service VS factory VS provider】---它们的区别,你知道么?

    在介绍AngularJS自定义服务之前,我们先来了解一下AngularJS~ 学过HTML的人都知道,HTML是一门很好的伪静态文本展示设计的声明式语言,但是,要构建WEB应用的话它就显得乏力了. 而 ...