发布时间:2025-06-24 17:48:44 作者:北方职教升学中心 阅读量:451
通过维护二叉树中每个节点的左子树所有值均小于它的值,右子树所有值均大于它的值的特性,二叉搜索树在插入、右孩子结点
而上面四种情况可以转化成下面的情况:
- 删除该结点且使被删除节点的双亲结点指向被删除节点的左孩子结点–直接删除
- 删除该结点且使被删除节点的双亲结点指向被删除结点的右孩子结点–直接删除
- 在它的右子树中寻找中序下的第一个结点(关键码最小),用它的值填补到被删除节点中,再来处理该结点的删除问题–替换法删除
这三种情况就是我们模拟实现删除的方法!
删除代码实现(示例):
boolErase(constK&key){Node*cur =_root;Node*parent =nullptr;while(cur){parent =cur;if(key >cur->_key){cur =cur->_right;}elseif(key <cur->_key){cur =cur->_left;}else{// 准备删除// 左为空if(cur->_left ==nullptr){// 当需要删除的是头节点时if(cur ==_root){_root =cur->_right;}else{if(cur ==parent->_left){parent->_left =cur->_right;}else{parent->_right =cur->_right;}}}// 右为空elseif(cur->_right ==nullptr){// 当需要删除的是头节点时if(cur ==_root){_root =cur->_left;}else{if(cur ==parent->_left){parent->_left =cur->_left;}else{parent->_right =cur->_left;}}}// 两边都不为空else{// 这里与外层不是同一块作用域,所以可以再次定义parent,不把parent定义为nullptr是因为//,可能不进入下面循环导致对parent空指针的使用Node*subLeft =cur->_right;// 定义右数节点Node*parent =cur;while(subLeft->_left){parent =subLeft;subLeft =subLeft->_left;}swap(cur->_key,subLeft->_key);// 替换右子树的最左节点if(subLeft ==parent->_left){// 最左节点肯定没有左孩子,所以让parent接管它的右子树parent->_left =subLeft->_right;}else{parent->_right =subLeft->_right;}deletesubLeft;}returntrue;}}returnfalse;}
🧩二叉搜索树默认成员函数
构造
BSTree()=default;// 显式地定义默认构造函数
拷贝构造
BSTree(constBSTree<K>&t){_root =Copy(t._root);}Node*Copy(Node*root){if(root ==nullptr){returnnullptr;}// 递归进行拷贝构造Node*newroot =newNode(root->_key);newroot->_left =Copy(root->_left);newroot->_right =Copy(root->_right);returnnewroot;}
赋值重载
BSTree<K>&operator=(BSTree<K>t){// 现代写法-> 直接调用swapswap(_root,t._root);return*this;}
析构
~BSTree(){Destory(_root);}voidDestory(Node*&_root){if(_root ==nullptr){return;}// 递归调用析构Destory(_root->_left);Destory(_root->_right);delete_root;_root ==nullptr;}
📜3. 二叉搜索树模拟实现(递归)
在进行递归操作的模拟实现时,一般都要传节点,进行多层的调用,因为我们都要定义两层
boolFindR(constK&key){return_FindR(_root,key);}boolInsertR(constK&key){return_InsertR(_root,key);}boolEraseR(constK&key){return_EraseR(_root,key);}
🌞插入
代码实现(示例):
bool_InsertR(Node*&_root,constK&key){// 递归出口if(_root ==nullptr){// 这里我们无需在进行对新节点的连接,因为我们是传引用传参,_root =newNode(key);returntrue;}if(key >_root->_key){return_InsertR(_root->_right,key);}elseif(key <_root->_key){return_InsertR(_root->_left,key);}else{returnfalse;}}
🌙查找
代码实现(示例):
bool_FindR(Node*_root,constK&key){if(_root ==nullptr){returnfalse;}if(key >_root->_key){return_FindR(_root->_right,key);}elseif(key <_root->_key){return_FindR(_root->_left,key);}else{returntrue;}}
⭐删除
代码实现(示例):
bool_EraseR(Node*&_root,constK&key){if(_root ==nullptr){returnfalse;}if(key >_root->_key){return_EraseR(_root->_right,key);}elseif(key <_root->_key){return_EraseR(_root->_left,key);}else{// 删除if(_root->_left ==nullptr){Node*del =_root;_root =_root->_right;deletedel;returntrue;}elseif(_root->_right ==nullptr){Node*del =_root;_root =_root->_left;deletedel;returntrue;}else{Node*subLeft =_root->_right;while(subLeft->_left){subLeft =subLeft->_left;}swap(_root->_key,subLeft->_key);// 让子树继续递归下去return_EraseR(_root->_right,key);}}}
📚4. 二叉搜索树的应用
🍁KV模型
KV模型:每一个关键码key,都有与之对应的值Value,即<Key, Value>的键值对。这些都需要我们付出大量的时间和精力去学习和实践。对于任何对编程和数据结构感兴趣的人来说,掌握二叉搜索树都是至关重要的一步
二叉搜索树不仅仅是一个简单的数据结构,它更是一种解决问题的方式和思维的体现。遍历以及操作的实现
让我们一起踏上学习二叉搜索树的旅程,探索它带来的无尽可能!
(本文重在二叉搜索树的模拟实现与理解)📒1. 二叉搜索树
🎩二叉搜索树概念
二叉搜索树又称二叉排序树,它或者是一棵空树,或者是具有以下性质的二叉树:
- 若它的左子树不为空,则左子树上所有节点的值都小于根节点的值
- 若它的右子树不为空,则右子树上所有节点的值都大于根节点的值
- 它的左右子树也分别为二叉搜索树
🎈二叉搜索树操作
首先,在二叉搜索树的操作中只支持插入,查找,删除,遍历,并不支持
修改
操作,因为在修改后谁也不能保证它依然是一棵二叉搜索树,二叉搜索树的时间复杂度范围在(O(logN)~O(N)
)在二叉搜索树的遍历中一般采用中序遍历:先遍历左子树,然后访问根节点,最后遍历右子树。我们将从二叉搜索树的基本概念出发,逐步深入到其性质、查找和删除操作中展现出了卓越的性能。构建、在BST中,中序遍历会按照升序访问所有节点
二叉搜索树示例
inta[]={8,3,1,10,6,4,7,14,13};
📕2. 二叉搜索树模拟实现
🧩二叉搜索树结构
二叉搜索树结构的和树形结构差不多,这意味着每个元素(通常称为节点)都有两个指针:一个指向前一个左子树,另一个指向右子树,因此我们需要单独再定义一个类来表示节点结构,每个节点再串联起来构成BST
(在模拟实现二叉搜索树时,不用定义命名空间,因为不会和库中发生冲突)
节点定义(示例):
template<classK>structBSTreeNode{BSTreeNode<K>*_left;BSTreeNode<K>*_right;K _key;BSTreeNode(constK&key =K()):_key(key),_left(nullptr),_right(nullptr){}};
BST定义(示例):
template<classK>classBSTree{typedefBSTreeNode<K>Node;public:// 构造函数等可能的其他成员函数... private:Node*_root =nullptr;};
🧩二叉搜索树操作
🌈插入
插入的具体过程如下:
- 树为空,则直接新增节点,赋值给root指针
- 树不空,按二叉搜索树性质查找插入位置,插入新节点
插入代码实现(示例):boolInsert(constK&key){// 当根为空时,直接插入if(_root ==nullptr){_root =newNode(key);returntrue;}// 定义parent是因为,在最后找到插入位置时,需要parent将节点进行连接Node*parent =nullptr;Node*cur =_root;while(cur){parent =cur;// 插入的值比cur位置大时,cur往右移动if(key >cur->_key){cur =cur->_right;}// 插入的值比cur位置小时,cur往左移动elseif(key <cur->_key){cur =cur->_left;}// 当插入的值和cur位置相等时,直接退出,因为二叉搜索树不允许有相同的元素else{returnfalse;}}// 将插入位置的新节点与二叉搜索树连接cur =newNode(key);if(parent->_key <key){parent->_right =cur;}else{parent->_left =cur;}returntrue;}
🌞遍历
在二叉搜索树的遍历上,我们依旧采用当初二叉树时的中序遍历,但是我们想要递归遍历就必须调用节点,这里我们要调用两层
遍历代码实现(示例):
voidInOrder(){_InOrder(_root);cout <<endl;}void_InOrder(Node*root){// 递归出口if(root ==nullptr){return;}_InOrder(root->_left);cout <<root->_key <<" ";_InOrder(root->_right);}
遍历比较简单奥,我们接着往下讲
🌙查找
二叉搜索树的查找
- 从根开始比较,查找,比根大则往右边走查找,比根小则往左边走查找
- 最多查找高度次,走到到空,还没找到,这个值不存在
查找代码实现(示例):
boolFind(constK&key){Node*cur =_root;while(cur){// 查找的值比cur大,cur往右移动if(key >cur->_key){cur =cur->_right;}// 查找的值比cur小,cur往左移动elseif(key <cur->_key){cur =cur->_left;}// 找到key,返回trueelse{returntrue;}}returnfalse;// 找不到key,返回false}
⭐删除
首先查找元素是否在二叉搜索树中,如果不存在,则返回, 否则要删除的结点可能分下面情况:
- 要删除的结点无孩子结点
- 要删除的结点只有左孩子结点
- 要删除的结点只有右孩子结点
- 要删除的结点有左、前方还有更多数据结构等待你去探索,更多算法等待你去实践
不要忘记持续学习和自我提升。它以其高效的数据检索能力和独特的树形结构,在计算机科学领域扮演着举足轻重的角色。它需要我们深入理解其性质、删除还是查找操作,都体现了其高效和灵活的特点
学习的道路永无止境。搜索二叉树以其独特的性质在数据检索领域展现了出色的性能,无论是插入、原理和算法实现。我们需要掌握如何构建一棵二叉搜索树,如何遍历它,以及如何在其中进行高效的查找、剩余");dict.insert("string","字符串");dict.insert("hahaha","哈哈");dict.insert("Eternity","永恒");dict.insert("sort","排序");dict.InOrder();string str;while(cin >>str){kv::BSTreeNode<string,string>*ret =dict.Find(str);if(ret){cout <<ret->_value <<endl;}else{cout <<"无此单词"<<endl;}}}
🔥计数
代码实现(示例):
intmain(){string arr[]={"苹果","西瓜","苹果","西瓜","苹果","苹果","西瓜","苹果","香蕉","苹果","香蕉"};kv::BSTree<string,int>countTree;for(auto&e :arr){kv::BSTreeNode<string,int>*ret =countTree.Find(e);if(ret ==nullptr){countTree.insert(e,1);}else{ret->_value++;}}countTree.InOrder();return0;}
🌄二叉树巩固知识
最后在这给大家推荐几道巩固二叉树的编程题
二叉树创建字符串
二叉树的分层遍历
找到树中两个指定节点的最近公共祖先
二叉树搜索树转换成排序双向链表
根据一棵树的前序遍历与中序遍历构造二叉树
二叉树中序遍历 ,非递归迭代实现📖5. 总结
经过我们一同对搜索二叉树的深入学习和探索,相信你已经对这种数据结构有了全面而深刻的理解。计算机科学是一个日新月异的领域,新的技术和算法不断涌现。只有保持对知识的渴望和追求,我们才能在这个领域中不断前行,让我们一起在学习的道路上不断前行!
希望本文能够为你提供有益的参考和启示,让我们一起在编程的道路上不断前行!
谢谢大家支持本篇到这里就结束了,祝大家天天开心!