前言:
本章我们将认识一种新的二叉树——搜索二叉树。这棵树有个神奇的功能就是会对数据自动排序且有着非常高的查找效率。搜索二叉树作为set、map
的基础结构,同样又是接下来将要学到的AVL
树以及红黑树的实现基础非常值得我们去深入学习~
叉搜索树本质上也是一种二叉树,只不过多了一个约束规则——
若一棵二叉树不为空,则:
所以构建一个搜索二叉树,只需要在插入结点时满足约束规则即可。
与二叉树相同,二叉搜索树由一个个结点链接而成。每个结点包含三个成员——
template <class K>
struct BSTreeNode
{
BSTreeNode<K>* _left; //左孩子
BSTreeNode<K>* _right; //右孩子
K _key; //键值
BSTreeNode(const K& key)
:_left(nullptr)
, _right(nullptr)
, _key(key)
{}
};
所以再定义出BSTNode
(Binary Search Tree简写)结构体——
template <class K>
class BSTree
{
typedef BSTreeNode<K> Node;
public:
// 成员函数的实现
// 插入、删除、查找...
private:
Node* _root = nullptr;
};
接着就是各种成员函数的实现了
搜索二叉树的查找比较简单而且更容易帮助我们理解搜索二叉树的性质,所以先从查找入手。
以上图为例,倘若我们要查找 7,具体的思路是这样的——
true
于是我们试着着手实现一个Find
函数
bool Find(const K& key)
{
Node* cur = _root;
while (cur)
{
if (cur->_key < key) // 大于则去右子树查找
cur = cur->_right;
else if (cur->_key > key) // 小于则去左子树查找
cur = cur->_left;
else
return true; // 找到返回true
}
return false; // 未找到返回false
}
理解了如何查找,插入也就非常简单。
还是以此图为例,倘若我们要插入 9 ,具体步骤为——
cur
的位置,并随时更新parent
cur
走到10的左节点的位置,即cur = nullptr
,循环结束patent = Node*(10)
new
一个结Node*(key)
并赋值给parent->_left
即可。bool Insert(const K& key)
{
// 如果是第一次插入,直接new一个新结点给_root
if (_root == nullptr)
{
_root = new Node(key);
return true;
}
Node* cur = _root; // cur用来定位插入的位置
Node* parent = cur; // 记录parent的父亲
while (cur)
{
if (cur->_key < key)
{
parent = cur;
cur = cur->_right;
}
else if(cur->_key > key)
{
parent = cur;
cur = cur->_left;
}
else
{
return false;
}
}
// 插入
cur = new Node(key);
// 插入时依旧要进行判断
if (parent->_key < key)
parent->_right = cur;
else
parent->_left = cur;
return true;
}
二叉搜索树的删除是最精华的部分。对与叶子节点,例如4、7、13,删除非常简单,只需将自身的位置替换为nullptr
即可。
如果要删除14或者10,也是比较简单的,因为10的左右子树只有一方为nullptr
(10的左子树为空),所以只需要载删除的时候让父结点接管自己不为空的子树即可。
倘若要删除6或者3,由于它们的左右子树都不为空,删除时无法将两个子树都交给父结点,情况就较为复杂。
所以此种情况,我们只能想办法请一个人来接替自己的位置,但是并不是谁来都能胜任这个位置的。这个接替者必须满足二叉搜索树的条件——左子树都比它小,右子树都比它大。那么这个接替者的人选只能有这两个——
例如,倘若要删除3,此时有两种做法都可行——
综上所述,删除操作共分为一下几种情况——
bool Erase(const K& key)
{
Node* cur = _root;
Node* parent = cur;
while (cur)
{
// 找到值为key的结点
if (cur->_key < key)
{
parent = cur;
cur = cur->_right;
}
else if (cur->_key > key)
{
parent = cur;
cur = cur->_left;
}
else // 找到了
{
// 删除
if (cur->_left == nullptr) // 1.左子树为空
{
if (cur == _root) // 根节点的删除
{
_root = cur->_right;
return true;
}
else
{
if (parent->_left == cur)
parent->_left = cur->_right;
else
parent->_right = cur->_right;
delete cur;
}
}
else if (cur->_right == nullptr) // 2.右子树为空
{
if (cur == _root) // 根节点的删除
{
_root = cur->_left;
return true;
}
else
{
if (parent->_left == cur)
parent->_left = cur->_left;
else
parent->_right = cur->_left;
delete cur;
}
}
else // 左右子树都不为空
{
// 找左子树的最大结点 或者 右子树的最小结点
Node* minRight = cur->_right;
Node* pminRight = cur;
while (minRight->_left)
{
pminRight = minRight;
minRight = minRight->_left;
}
cur->_key = minRight->_key; // 替换
if (pminRight->_left == minRight)
{
pminRight->_left = minRight->_right;
}
else
{
pminRight->_right = minRight->_right;
}
delete minRight;
}
return true;
}
}
return false;
}
二叉搜索树的遍历非常简单,就是之前学习过的二叉树的中序遍历。
void InOrder()
{
_InOrder(_root);
cout << endl;
}
void _InOrder(Node* root)
{
if (root == nullptr)
return;
_InOrder(root->_left);
cout << root->_key << ' ';
_InOrder(root->_right);
}
注:由于调用函数时C++
封装的特性,需设计两个函数,InOrder
接口对外提供,_InOrder
不对外提供。
//采用前序遍历拷贝构造
BSTree(const BSTree<K>& t)
{
_root = _Copy(t._root);
}
Node* _Copy(Node* root)
{
if (root ==nullptr)
{
return nullptr;
}
Node* CopyRoot = new Node(root->_key);
CopyRoot->_left = _Copy(root->_left);
CopyRoot->_right = _Copy(root->_right);
return CopyRoot;
}
//采用后序遍历的方式来删除,从下往上删。
void _Destroy(Node*& root)
{
if (root == nullptr)
{
return;
}
_Destroy(root->_left);
_Destroy(root->_right);
delete root;
root = nullptr;
}
~BSTree()
{
_Destroy(_root);
}
因为拷贝构造二叉搜索树时要保证树的结构与原来树的结构一致,因此采用前序遍历进行拷贝构造。
但如果写了拷贝构造之后编译器就不会生成默认的构造函数了,因为拷贝构造也属于构造,因此可以利用一下C++11
的特性,强制编译器生成一个默认的拷贝构造
//强制编译器生成一个默认的拷贝构造
BSTree() = default;
对于搜索二叉树来说,上面实现的非递归版本是比递归版本更优的。此处的递归实现完全属于多余了,但是作为拓展内容看一看也未尝不可。
bool FindR(const K& key)
{
return _FindR(_root, key);
}
bool _FindR(Node* root, const K& key)
{
if (root == nullptr)
return false;
if (root->_key == key)
return true;
if (root->_key > key)
_FindR(root->_left, key);
else
_FindR(root->_right, key);
}
bool InsertR(const K& key)
{
return _EraseR(_root, key);
}
bool _InsertR(Node*& root, const K& key)
{
if (root == nullptr)
{
root = new Node(key);
return true;
}
if (root->_key < key)
return _InsertR(root->_right, key);
else if (root->_key > key)
return _InsertR(root->_left, key);
else
return false;
}
bool EraseR(const K& key)
{
return _EraseR(_root, key);
}
bool _EraseR(Node*& root, const K& key)
{
if (root == nullptr)return false;
if (root->_key < key)
return _EraseR(root->_right, key);
else if(root->_key>key)
return _EraseR(root->_left, key);
else
{
Node* del = root;
//1.右为空
if (root->_right == nullptr)
root = root->_left;
//2.左为空
else if (root->_left)
root = root->_right;
//3.左右都不为空
else
{
//找左子树最大节点交换
Node* maxleft = root->_left;
while (maxleft->_right)
maxleft = maxleft->_right;
//找到后先交换要删除的值与左子树最大节点的值
swap(root->_key, maxleft->_key);
//再递归到左子树中去删除
return _EraseR(root->_left, key);
}
delete del;
return true;
}
}
插入和删除操作都必须先查找,查找效率代表了二叉搜索树中各个操作的性能。
对有n个结点的二叉搜索树,若每个元素查找的概率相等,则二叉搜索树平均查找长度是结点在二
叉搜索树的深度的函数,即结点越深,则比较次数越多。
但对于同一个关键码集合,如果各关键码插入的次序不同,可能得到不同结构的二叉搜索树:
如果退化成单支树,二叉搜索树的性能就失去了。那能否进行改进,不论按照什么次序插 入关键码,二叉搜索树的性能都能达到最优? 那么我们后续章节学习的AVL树和红黑树就可以上场了。
K
模型:K
模型即只有key
作为关键码,结构中只需要存储Key
即可,关键码即为需要搜索到的值。word
,判断该单词是否拼写正确,具体方式如下:
key
,构建一棵二叉搜索树KV
模型:每一个关键码key
,都有与之对应的值Value
,即
的键值对。该种方式在现实生活中非常常见:
就构成一种键值对
就构成一种键值对namespace dianxia
{
template <class K>
struct BSTreeNode
{
BSTreeNode<K>* _left;
BSTreeNode<K>* _right;
K _key;
BSTreeNode(const K& key)
:_left(nullptr)
, _right(nullptr)
, _key(key)
{}
};
template <class K>
class BSTree
{
typedef BSTreeNode<K> Node;
public:
//强制编译器生成构造函数
BSTree() = default;
//拷贝构造
BSTree(const BSTree<K>& t)
{
_root = Copy(t._root);
}
//赋值重载
BSTree<k>& operator=(BSTree<K> t)
{
swap(_root, t._root);
return *this;
}
~BSTree()
{
Destroy(_root);
}
//插入节点
bool Insert(const K& key)
{
//直接插入根
if (_root == nullptr)
{
_root = new Node(key);
return true;
}
Node* parent = nullptr;
Node* cur = _root;
while (cur)
{
if (cur->_key < key)
{
parent = cur;
cur = cur->_right;
}
else if (cur->_key > key)
{
parent = cur;
cur = cur->_left;
}
else
{
returen false;
}
}
cur = new Node(key);
if (parent->_key < key)
{
parent->_right = cur;
}
else
{
parent->_left = cur;
}
return true;
}
bool Find(const K& key)
{
Node* cur = _root;
while (cur)
{
if (cur->_key < key)
{
cur = cur->_right;
}
else if (cur->_key > key)
{
cur = cur->_left;
}
else
{
returen true;
}
}
return false;
}
bool Erase(const K& key)
{
Node* parent = nullptr;
Node* cur = _root;
while (cur)
{
if (cur->_key < key)
{
parent = cur;
cur = cur->_right;
}
else if (cur->_key > key)
{
parent = cur;
cur = cur->_left;
}
else
{
//1.左为空
if (cur->_left == nullptr)
{
if (cur == _root)
{
_root = cur->_right;
}
else
{
if (parent->_left == cur)
{
parent->_left = cur->_right;
}
else
{
parent->_right == cur->_right;
}
}
delete cur;
}
//2.右为空
else if (cur->_right == nullptr)
{
if (cur == _root)
{
_root = cur->_left;
}
else
{
if (parent->_left == cur)
{
parent->_left = cur->_left;
}
else
{
parent->_right == cur->_left;
}
}
delete cur;
}
//3.左右都不为空,找右树最小节点或左树最大节点替代cur
else
{
Node* pminRight = cur;
Node* minRight = cur->_right;
while (minRight->_left)
{
pminRight = minRight;
minRight = minRight->_left;
}
cur->_key = minRight->_key;
if (pminRight->_left == minRight)
{
pminRight->_left = minRight->_right;
}
else
{
pminRight->_right = minRight->_right;
}
delete minRight
}
return true;
}
}
return false;
}
//递归版
bool FindR(const K& key)
{
return _FindR(_root, key);
}
bool InsertR(const K& key)
{
return _InsertR(_root, key);
}
bool Erase(const K& key)
{
return _Erase(_root, key);
}
bool Insorder()
{
_Insorder(_root);
cout << endl;
}
protected:
Node* Copy(Node* root)
{
if (root == nullptr)
return nullptr;
Node* newRoot = new Node(root->_key);
newRoot->_left = Copy(root->_left);
newRoot->_right = Copy(root->_right);
return newRoot;
}
void Destroy(Node*& root)
{
if (root == nullptr)
return;
Destroy(root->_left);
Destroy(root->_right);
delete root;
root = nullptr;
}
bool _FindR(Node* root, const K& key)
{
if (root == nullptr)
return false;
if (root->_key == key)
return true;
if (root->_key < key)
return _FindR(root->_right, key);
else
return _FindR(root->_left, key);
}
bool _InsertR(Node*& root, const K& key)
{
if (root == nullptr)
{
root = new Node(key);
return true;
}
if (root->_key < key)
{
return _InsertR(root->_right, key);
}
else if (root->_key > key)
{
return _InsertR(root->_left, key);
}
else
{
return false;
}
}
bool _EraseR(Node*& root, const K& key)
{
if (root == nullptr)return false;
if (root->_key < key)
return _EraseR(root->_right, key);
else if(root->_key>key)
return _EraseR(root->_left, key);
else
{
Node* del = root;
//1.右为空
if (root->_right == nullptr)
root = root->_left;
//2.左为空
else if (root->_left)
root = root->_right;
//3.左右都不为空
else
{
//找左子树最大节点交换
Node* maxleft = root->_left;
while (maxleft->_right)
maxleft = maxleft->_right;
//找到后先交换要删除的值与左子树最大节点的值
swap(root->_key, maxleft->_key);
//再递归到左子树中去删除
return _EraseR(root->_left, key);
}
delete del;
return true;
}
}
private:
Node* _root;
};
}
本文到此结束,码文不易,还请多多支持!!!