二叉搜索树,又称为二叉排序树,二叉查找树,它满足如下四点性质:
1)空树是二叉搜索树;
2)若它的左子树不为空,则左子树上所有结点的值均小于它根结点的值;
3)若它的右子树不为空,则右子树上所有结点的值均大于它根结点的值;
4)它的左右子树均为二叉搜索树;
如上图所示:二叉搜索树的任何一棵子树,它的根结点的值一定大于左子树所有结点的值,且一定小于右子树所有结点的值。如果对二叉搜索树进行中序遍历,我们可以发现,得到的序列是一个递增序列,上述的遍历结果为[1,2,3,4,5,6,7,8]。
如果要查找4,只需要从根结点比较查找3次就能找到,可以显著提高搜索的速度。
在二叉搜索树中查找某个数是否存在,存在返回 true,不存在返回 false。
对于要查找的数 val ,从根结点出发,总共四种情况依次判断:
1)若二叉搜索树为空树,直接返回 false;
2) val 的值 等于 树根结点的值,则直接返回 true;
3) val 的值 小于 树根结点的值,说明 val 对应的结点不在根结点,也不在右子树上,需要在左子树上查找,递归返回左子树的查找结果;
4) val 的值 大于 树根结点的值,说明 val 对应的结点不在根结点,也不在左子树上,需要在右子树上查找,递归返回右子树的查找结果;
① 结点源码
struct TreeNode {
int val;
struct TreeNode *left;
struct TreeNode *right;
TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
};
② 查找算法源码 (深度优先,递归查找)
bool BSTFind(TreeNode* root, int val)
{
if (root == nullptr) {
return false;
}
if (root->val == val) {
return true;
}
if (val < root->val) {
return BSTFind(root->left, val);
}
else {
return BSTFind(root->right, val);
}
}
将给定的值 val 生成结点后,插入到树上的某个位置,并且保持这棵树还是二叉搜索树。对于要插入的值 val ,从根结点出发,总共四种情况依次判断:
1)若为空树,则创建一个值为 val 的结点并且返回根结点;
2) val 的值 等于 树根结点的值,无须执行插入,直接返回根结点;
3) val 的值 小于 树根结点的值,那么插入位置一定在 左子树,递归执行插入左子树的过程,并且返回插入结果作为新的左子树;
4) val 的值 大于 树根结点的值,那么插入位置一定在 右子树,递归执行插入右子树的过程,并且返回插入结果作为新的右子树;
TreeNode* BSTInsert(TreeNode* root, int val) {
if (root == nullptr) {
root = new TreeNode(val);
return root;
}
if (val == root->val) {
return root;
}
if (val < root->val) {
root->left = BSTInsert(root->left, val);
}
else {
root->right = BSTInsert(root->right, val);
}
return root;
}
删除值为 val 结点,从根结点出发,总共四种情况依次判断:
1)空树,不存在结点直接返回空树;
2) val 的值 小于 树根结点的值,则需要删除的结点一定不在右子树上,递归调用删除左子树的对应结点;
3) val 的值 大于 树根结点的值,则需要删除的结点一定不在左子树上,递归调用删除右子树的对应结点;
4) val 的值 等于 树根结点的值,相当于是要删除根结点,这时候又要分三种情况:
由上述删除算法原理可知,删除结点之前可能还需要找最小结点,所以需要定义查找最小结点接口。
TreeNode* BSTFindMin(TreeNode* root) {
while (root && root->left)
{
root = root->left;
}
return root;
}
查找根为 root ,值最小的那个结点的值,根据二叉搜索树的性质,如果左子树存在,则必然存在更小的值,递归搜索左子树,且最小值结点为叶子结点;如果左子树不存在,则根结点的值必然最小,直接返回。
删除根结点,并返回新根结点
//删除根结点并返回新根结点
TreeNode* Delete(TreeNode* root) {
TreeNode* delNode=nullptr, * retNode=nullptr;
if (!root) { //空树,直接返回空
return nullptr;
}
if (root->left == nullptr) { //只有右子树(包含单结点),删除根空间后返回右子树根结点
delNode = root;
retNode = root->right;
delete delNode;
}
else if (root->right == nullptr) { //只有左子树,删除根空间后返回左子树根结点
delNode = root;
retNode = root->left;
delete delNode;
}
else {
retNode = root;
TreeNode* cur = root->right;
TreeNode* pcur = root;
while (cur->left)
{
pcur = cur;
cur = cur->left;
}
delNode = cur;
retNode->val = cur->val;
if (pcur->left == cur) {//右子树的最小值在右子树的左子树上
pcur->left = cur->right;
}
else {//右子树的最小值为右子树的根
pcur->right = cur->right;
}
delete delNode;
}
return retNode;
}
删除指定值的结点
//删除指定结点
TreeNode* BSTDelete(TreeNode* root, int val) {
if (nullptr == root) {
return nullptr;
}
if (val == root->val) {
return Delete(root);
}
else if (val < root->val) {
root->left = BSTDelete(root->left, val);
}
else if (val > root->val) {
root->right = BSTDelete(root->right, val);
}
return root;
}