C++《二叉搜索树》

news2025/1/15 6:57:49

在初阶数据结构中我学习了树基础的概念以及了解了顺序结构的二叉树——堆和链式结构二叉树该如何实现,那么接下来我们将进一步的学习二叉树,在此会先后学习到二叉搜索树、AVL树、红黑树;通过这些的学习将让我们更易于理解后面set、map、哈希等的使用以及对底层结构的了解。在此先本篇中我们将了解二次搜索树的概念以及实现二叉搜索树插入、删除等的操作,在了解了这些之后相信在下一篇的set和map的学习你将轻松许多,接下来就开始本篇的学习吧!!!


 1.二叉搜索树的概念

二叉搜索树又称二叉排序树,它或者是⼀棵空树,或者是具有以下性质的二叉树:

• 若它的左子树不为空,则左子树上所有结点的值都小于等于根结点的值
• 若它的右子树不为空,则右子树上所有结点的值都
大于等于根结点的值
• 它的左右子树也分别为二叉搜索树
• 二叉搜索树中可以支持插入相等的值,也可以不支持插入相等的值,具体看使用场景定义,后续我们学习map/set/multimap/multiset系列容器底层就是二叉搜索树,其中map/set不支持插入相等值,multimap/multiset支持插入相等值

例如以下左边图示的就是不支持插入相等值得二叉搜索树,右边就是支持插入相等值得二叉搜索树

2. 二叉搜索树的性能分析 

最好得情况下,在此在二叉搜索树当中最为完全二叉树(或者接近完全二叉树),其高度为: O(log2 N)

例如以下示例:

最差情况下,在此⼆叉搜索树退化为单支树(或者类似单支),其高度为:
O(\frac{N}{2})

例如以下示例:

那么通过以上对二叉树最好和最坏情况的分析就可以得出综合而言二叉搜索树增删查改时间复杂度为: O(N)

那么通过以上的分析可以看出二叉搜索树这样的效率显然无法满足我们需求的,我们后续需要继续讲解二叉搜索树的变形,平衡二叉搜索树AVL树和红黑树,才能适用于我们在内存中存储和搜索数据。


在此你可能会想到在二叉搜索树中查找数据不就类似与二分查找吗,那么直接使用之前学习的二分查找不就可以了吗,二分查找的效率相比二叉搜索树还更好,那为什么还要学习了解二叉搜索树呢?
在此二分查找也可以实现 O(logN) 级别的查找效率,但是二分查找有两大缺陷:

1. 需要存储在支持下标随机访问的结构中,并且有序。

在使用到二分查找示我们使用的是数据对应的下标来实现查找,这就使得当被查找的一系列数据不是存储在数组里时就需要先将数据都存储在数组当中并且还要将数据排序成升序,在这个过程中就会有时间和空间上的损耗了 

2. 插入和删除数据效率很低,因为存储在下标随机访问的结构中,插入和删除数据⼀般需要挪动数据。

在二分查找当中以上提到的缺点其实还不是最致命的,最要命的是当一组数据已经存储在数组当中时并且已经排序好时,如果之后我们要在这组数据当中再插入新的元素或者是要将原数据中的一个元素删除,那么通过之前顺序表的学习我们就知道每插入或者是删除一个元素时间复杂度都为O(N),若多次进行操作这就使得时间复杂度非常高了

int a[] = {8, 3, 1, 10, 6, 4, 7, 14, 13};


通过以上的分析这里也就体现出了平衡二叉搜索树的价值,在插入元素或者是删除元素都相比使用二分查找要有优势。

3. 二叉搜索树的插入

3.1插入分析

在此在二叉搜索树当中插入新的节点就要按照以下步骤进行分析:
1. 树为空,则直接新增结点,赋值给root指针
2. 树不空,按二叉搜索树性质,插入值比当前结点大往右走,插入值比当前结点小往左走,找到空位置,插入新结点。
3. 如果支持插入相等的值,插入值跟当前结点相等的值可以往右走,也可以往左走,找到空位置,插入新结点。(要注意的是要保持逻辑⼀致性,插入相等的值不要⼀会往右走,⼀会往左走

例如以下示例:
假设我们要将以下的数组元素依次插入到二叉搜索树当中

int a[] = {8, 3, 1, 10, 6, 4, 7, 14, 13};

 所以元素插入完之后二叉搜索树的形式就如下所示:
 

这时如果我们要再插入值为16的元素在二叉搜索树中过程图就如下所示:

 

 再插入值为3的元素在二叉搜索树中过程图就如下所示:
 

 

3.2 插入代码实现 

在实现二叉搜索树的插入代码之前我们先要来实现二叉搜索树大体的结构代码

在此我们先创建一个BSTree.h的头文件在该文件当中来实现二叉搜索树的结构以及各个功能,再创建一个test.cpp的文件用于测试我们实现的二叉搜索树的各个功能是否能满足要求

实现了文件的创建之后接下来就来实现二叉搜索树的大体结构。在此由于二叉搜索树是由各个节点构成的,那么和之前实现链式结构的二叉树一样先要实现表示节点的结构体

 

#include<iostream>
using namespace std;


template<class K>
struct BSTreeNode
{
	K _key;
	BSTreeNode<K>* left;
	BSTreeNode<K>* right;
	BSTreeNode(const K& key)
		:_key(key)
		,left(nullptr)
		,right(nullptr)
	{

	}
};

在此就创建一个结构体BSTreeNode来表示二叉树内节点,在节点当中有三个变量分别是_key表示节点内的数据、_left存储该节点左孩子节点的指针、_right存储该节点右孩子节点的指针。并且将该结构体实现成模板这样就可以支持节点内的元素是任意类型的数据,还有在结构体当中还实现了构造函数。

实现了节点的结构体之后接下来就可以来实现表示二叉搜索树的类,在此我们将类命名为BSTree,实现的是模板类,实现的大体结构如下所示:

template<class K>
class BSTree
{
	typedef BSTreeNode<K> Node;

public:
    //使用编译器生成的默认构造函数
    BSTree() = default;
    //实现各种功能的成员函数……
	

private:
    //头节点
	Node* _root = nullptr;

};

完成了以上操作接下来就可以来实现插入函数的代码了

注:以下实现的插入函数是数据不支持冗余的情况也就是二叉树当中不支持插入相等的值

bool Insert(const K& key)
{
	//当根节点为空时
	if (_root == nullptr)
	{
		_root = new Node(key);
		return true;
	}

	Node* cur = _root;
	//节点的父节点
	Node* parentcur = nullptr;
	while (cur)
	{
		//当key小于当前节点的值
		if (cur->_key < key)
		{
			parentcur = cur;
			cur = cur->right;
		}
		当key大于当前节点的值
		else if (cur->_key > key)
		{
			parentcur = cur;
			cur = cur->left;

		}
		//当key等于当前节点的值
		else
		{
			return false;
		}

	}
	cur = new Node(key);
	//当新节点内的值大于父节点内的值时
	if (parentcur->_key  <  key)
	{
		parentcur->right = cur;
	}
	//当新节点内的值小于父节点内的值时
	else
	{
		parentcur->left = cur;
	}
	return true;

}

4. 二叉搜索树的查找

4.1查找分析

在二叉搜索树中查找节点就要按照以下步骤进行分析:

1. 从根开始比较,查找x,x比根的值大则往右边走查找,x比根值小则往左边走查找。
2. 最多查找高度次,走到到空,还没找到,这个值不存在。
3. 如果不支持插入相等的值,找到x即可返回
4. 如果支持持插入相等的值,意味着有多个x存在,⼀般要求查找中序的第⼀个x。

例如以下示例: 

当要查找3时,要找到1的右孩子的那个3值返回 

 

4.2 查找代码实现 

在进行了二叉搜索树的节点查找的分析之后接下来我们就来实现查找代码

注:以下实现的查找函数是数据不支持冗余的情况也就是二叉树当中不支持插入相等的值 

Node* Find(const K& key)
{
	//当根结点为空时
	if (_root == nullptr)
	{
		return nullptr;
	}
	Node* cur = _root;
	Node* parentcur = nullptr;
	while (cur)
	{
		//当key大于当前节点的值
		if (cur->_key < key)
		{
			parentcur = cur;
			cur = cur->right;
		}
		//当key小于当前节点的值
		else if (cur->_key > key)
		{
			parentcur = cur;
			cur = cur->left;
		}
		//当key等于当前节点的值
		else
		{
			return cur;
		}
	}
	//当前二叉树中找不到值为key的节点
	return nullptr;
}

5. 二叉搜索树的删除 

5.1 删除分析

在二叉搜索树当中节点的删除相比插入和查找就复杂一些了,在此会出现以下的多种情况接下来就来一一分析

在此要删除的节点会有以下的四种情况:

1.要删除结点N左右孩子均为空

当要删除节点的节点左右孩子节点都为空时就需要把N结点的父亲对应孩子指针指向空,直接删除N结点

例如以下示例:

在以上二叉搜索树当中要删除值为1的节点就需要将节点1删除之后再将其父节点的左孩子节点指向空

2. 要删除的结点N左孩子位空,右孩子结点不为空

当要删除节点的节点左孩子节点为空时就需要把N结点的父亲对应孩子指针指向N的右孩子,之后直接删除N结点

例如以下示例:


 

在以上二叉搜索树当中要删除值为10的节点就需要将其父节点的左孩子变为原节点的右孩子节点之后再将节点10删除

 3.要删除的结点N右孩子位空,左孩子结点不为空

当要删除节点的节点右孩子节点为空时就需要把N结点的父亲对应孩子指针指向N的左孩子,之后直接删除N结点

例如以下示例:

 

在以上二叉搜索树当中要删除值为14的节点就需要将其父节点的左孩子变为原节点的左孩子节点之后再将节点14删除

4. 要删除的结点N左右孩子结点均不为空

当要删除的节点左右孩子节点都不为空时,这是就不能像以上的情况一样简单的改变要删除节点的父节点指针,这时由于无法直接删除N结点,这是因为N的两个孩⼦无处安放,只能用替换法删除。在此根据二叉搜索树的性质就需要找N左子树的值最大结点R(最右结点)或者N右子树的值最小结点R(最左结点)替代N,因为这两个结点中任意⼀个,放到N的位置,都满足⼆叉搜索树的规则。替代N的意思就是N和R的两个结点的值交换,转而变成删除R结点,R结点符合情况2或情况3,可以直接删除。 

例如以下示例:
 

在以上二叉搜索树当中要删除节点值为8的节点和值为3的节点,由于这两个节点都是左右孩子节点都不为空的节点,因此要删除值为8的节点就需要找到其左子树最右的节点或者是右子树最左的节点(在此我们是找右子树最左的节点)

之后交换要删除的节点和找出的节点的值

 

最后删除交换之后值为8的节点即可

 

注:要删除值为3的节点和以上的方式也类型在此就不再细致的讲解

 

5.2 删除代码实现

bool Erase(const K& key)
{
	//当根节点为空时
	if (_root == nullptr)
	{
		return false;
	}
	//当前节点
	Node* cur = _root;
	//当前节点的父节点
	Node* parentcur = _root;
	while (cur)
	{
		//当key的值大于当前节点的值
		if (cur->_key < key)
		{
			parentcur = cur;
			cur = cur->right;
		}
		//当key的值大于当前节点的值
		else if (cur->_key > key)
		{
			parentcur = cur;
			cur = cur->left;
		}
		//当key的值等于当前节点的值
		else
		{
			//当要删除的节点左孩子节点为空
			if (cur->left==nullptr)
			{
				//当要删除的节点为根结点时直接改变_root指针的指向,之后再将原根结点释放
				if (cur == _root)
				{
					Node* Right = cur->right;
					delete cur;
					_root = Right;
				}
				else
				{
					//当要删除的节点cur是其父节点的左节点时
					if (cur == parentcur->left)
					{
						parentcur->left = cur->right;
						delete cur;
					}
					//当要删除的节点cur是其父节点的右节点时
					else if (cur == parentcur->right)
					{
						parentcur->right = cur->right;
						delete cur;
					}
				}
				
				return true;

			}
			//当要删除的节点右孩子节点为空
			else if (cur->right==nullptr)
			{
				//当要删除的节点为根结点时直接改变_root指针的指向,之后再将原根结点释放
				if (cur == _root)
				{
					Node* Left = cur->left;
					delete cur;
					_root = Left;
				}
				else
				{
					//当要删除的节点cur是其父节点的左节点时
					if (cur == parentcur->left)
					{
						parentcur->left = cur->left;
						delete cur;
					}
					//当要删除的节点cur是其父节点的右节点时
					else if (cur == parentcur->right)
					{
						parentcur->right = cur->left;
						delete cur;
					}
				}
				return true;
				
			}
			//当要删除的节点左右孩子节点都为空
			else
			{
				Node* minrightParent = cur;
				Node* minright = cur->right;
				//找出当前节点右子树中的最左节点
				while (minright->left)
				{
					minrightParent = minright;
					minright = minright->left;
				}
				cur->_key = minright->_key;
				//当最左节点为其父节点的左节点时
				if (minrightParent->left == minright)
				{
					minrightParent->left = minright->right;
					
				}
				//当最左节点为其父节点的右节点时
				else 
				{
					minrightParent->right = minright->right;
					
				}
				delete minright;
				return true;	

			}

		}
	}

	//当找不到要删除的节点就返回false
	return false;

}

 6. 二叉搜索树遍历

由于二叉搜索树的性质一棵二叉搜索树中序遍历输出的结果就是递增的,那么接下来我们就试着来实现中序遍历的代码

void Inorder()
{
	_Inorder(_root);
	cout << endl;
}



void _Inorder(Node* root)
{
	if (root == nullptr)
	{
		return;
	}
	_Inorder(root->left);
	cout << root->_key <<" ";
	_Inorder(root->right);
}

在此我们实现的中序遍历代码如上所示,那么这时你可能就会有疑问了,为什么要实现两个中序遍历的函数,不是直接使用一个函数就可以满足要求了吗?

在此要考虑到的是在BsTree类以外用户是无法得到二叉搜索树的根节点的,但是在调用中序遍历的函数根据之前我们使用递归的方式是需要一开始就需要将二叉树的根结点作为中序遍历函数的参数的。因此为了解决该问题就再在BSTree类内实现一个函数来调用中序遍历的成员函数,由于是在类的内部在此是可以访问私有的成员变量的,在类外部用户要使用中序遍历时就只需要调用无参的成员函数Inorder就可以得到二叉搜索树中序遍历的结果了。

7. 二叉搜索树完整代码

#include<iostream>
using namespace std;


template<class K>
struct BSTreeNode
{
	K _key;
	BSTreeNode<K>* left;
	BSTreeNode<K>* right;
	BSTreeNode(const K& key)
		:_key(key)
		, left(nullptr)
		, right(nullptr)
	{

	}
};

template<class K>
class BSTree
{
	typedef BSTreeNode<K> Node;

public:
	//使用编译器生成的默认构造函数
	BSTree() = default;

	bool Insert(const K& key)
	{
		//当根节点为空时
		if (_root == nullptr)
		{
			_root = new Node(key);
			return true;
		}

		Node* cur = _root;
		//节点的父节点
		Node* parentcur = nullptr;
		while (cur)
		{
			//当key小于当前节点的值
			if (cur->_key < key)
			{
				parentcur = cur;
				cur = cur->right;
			}
			当key大于当前节点的值
			else if (cur->_key > key)
			{
				parentcur = cur;
				cur = cur->left;

			}
			//当key等于当前节点的值
			else
			{
				return false;
			}

		}
		cur = new Node(key);
		//当新节点内的值大于父节点内的值时
		if (parentcur->_key < key)
		{
			parentcur->right = cur;
		}
		//当新节点内的值小于父节点内的值时
		else
		{
			parentcur->left = cur;
		}
		return true;

	}

	Node* Find(const K& key)
	{
		//当根结点为空时
		if (_root == nullptr)
		{
			return nullptr;
		}
		Node* cur = _root;
		Node* parentcur = nullptr;
		while (cur)
		{
			//当key大于当前节点的值
			if (cur->_key < key)
			{
				parentcur = cur;
				cur = cur->right;
			}
			//当key小于当前节点的值
			else if (cur->_key > key)
			{
				parentcur = cur;
				cur = cur->left;
			}
			//当key等于当前节点的值
			else
			{
				return cur;
			}
		}
		//当前二叉树中找不到值为key的节点
		return nullptr;
	}


	bool Erase(const K& key)
	{
		//当根节点为空时
		if (_root == nullptr)
		{
			return false;
		}
		//当前节点
		Node* cur = _root;
		//当前节点的父节点
		Node* parentcur = _root;
		while (cur)
		{
			//当key的值大于当前节点的值
			if (cur->_key < key)
			{
				parentcur = cur;
				cur = cur->right;
			}
			//当key的值大于当前节点的值
			else if (cur->_key > key)
			{
				parentcur = cur;
				cur = cur->left;
			}
			//当key的值等于当前节点的值
			else
			{
				//当要删除的节点左孩子节点为空
				if (cur->left == nullptr)
				{
					//当要删除的节点为根结点时直接改变_root指针的指向,之后再将原根结点释放
					if (cur == _root)
					{
						Node* Right = cur->right;
						delete cur;
						_root = Right;
					}
					else
					{
						//当要删除的节点cur是其父节点的左节点时
						if (cur == parentcur->left)
						{
							parentcur->left = cur->right;
							delete cur;
						}
						//当要删除的节点cur是其父节点的右节点时
						else if (cur == parentcur->right)
						{
							parentcur->right = cur->right;
							delete cur;
						}
					}

					return true;

				}
				//当要删除的节点右孩子节点为空
				else if (cur->right == nullptr)
				{
					//当要删除的节点为根结点时直接改变_root指针的指向,之后再将原根结点释放
					if (cur == _root)
					{
						Node* Left = cur->left;
						delete cur;
						_root = Left;
					}
					else
					{
						//当要删除的节点cur是其父节点的左节点时
						if (cur == parentcur->left)
						{
							parentcur->left = cur->left;
							delete cur;
						}
						//当要删除的节点cur是其父节点的右节点时
						else if (cur == parentcur->right)
						{
							parentcur->right = cur->left;
							delete cur;
						}
					}
					return true;

				}
				//当要删除的节点左右孩子节点都为空
				else
				{
					Node* minrightParent = cur;
					Node* minright = cur->right;
					//找出当前节点右子树中的最左节点
					while (minright->left)
					{
						minrightParent = minright;
						minright = minright->left;
					}
					cur->_key = minright->_key;
					//当最左节点为其父节点的左节点时
					if (minrightParent->left == minright)
					{
						minrightParent->left = minright->right;

					}
					//当最左节点为其父节点的右节点时
					else
					{
						minrightParent->right = minright->right;

					}
					delete minright;
					return true;

				}

			}
		}

		//当找不到要删除的节点就返回false
		return false;

	}

	void Inorder()
	{
		_Inorder(_root);
		cout << endl;
	}




private:
	//头节点
	Node* _root = nullptr;

	void _Inorder(Node* root)
	{
		if (root == nullptr)
		{
			return;
		}
		_Inorder(root->left);
		cout << root->_key << " ";
		_Inorder(root->right);
	}

};

8.  二叉搜索树key和key/value使用场景 

8.1 key搜索场景

key搜索场景的形式如下所示:

只有key作为关键码,结构中只需要存储key即可,关键码即为需要搜索到的值,搜索场景只需要判断key在不在。key的搜索场景实现的二叉树搜索树支持增删查,但是不支持修改,修改key破坏搜索树结构了。

在以上我们了解实现的二叉搜索树其实是适用于key的场景,那么接下来就来看看那么场景是属于key的场景

场景1:小区无人值守车库,小区车库买了车位的业主车才能进小区,那么物业会把买了车位的业主的车牌号录入后台系统,⻋辆进入时扫描⻋牌在不在系统中,在则抬杆,不在则提示非本小区⻋辆,无法进入。

 

场景2:检查⼀篇英文文章单词拼写是否正确,将词库中所有单词放入二叉搜索树,读取文章中的单词,查找是否在二叉搜索树中,不在则波浪线标红提示。 

8.2 key/value搜索场景

key/value搜索场景的形式如下所示:

每⼀个关键码key,都有与之对应的值value,value可以任意类型对象。树的结构中(结点)除了需要存储key还要存储对应的value,增/删/查还是以key为关键字走二叉搜索树的规则进行比较,可以快速查找到key对应的value。key/value的搜索场景实现的二叉树搜索树支持修改,但是不支持修改key,修改key破坏搜索树结构了,可以修改value。

接下来就来将以上我们实现的key二叉搜索树修改为key/value⼆叉搜索树代码,实现代码如下所示:

#include<iostream>
using namespace std;


template<class K,class V>
struct BSTreeNode
{
	K _key;
    V _value;
	BSTreeNode<K ,V>* left;
	BSTreeNode<K ,V>* right;
	BSTreeNode(const K& key, const V& value)
		:_key(key)
        ;_value(value)
		, left(nullptr)
		, right(nullptr)
	{

	}
};

template<class K,class V>
class BSTree
{
	typedef BSTreeNode<K,V> Node;

public:
	//使用编译器生成的默认构造函数
	BSTree() = default;

	bool Insert(const K& key, const V& value)
	{
		//当根节点为空时
		if (_root == nullptr)
		{
			_root = new Node(key,value);
			return true;
		}

		Node* cur = _root;
		//节点的父节点
		Node* parentcur = nullptr;
		while (cur)
		{
			//当key小于当前节点的值
			if (cur->_key < key)
			{
				parentcur = cur;
				cur = cur->right;
			}
			//当key大于当前节点的值
			else if (cur->_key > key)
			{
				parentcur = cur;
				cur = cur->left;

			}
			//当key等于当前节点的值
			else
			{
				return false;
			}

		}
		cur = new Node(key, value);
		//当新节点内的值大于父节点内的值时
		if (parentcur->_key < key)
		{
			parentcur->right = cur;
		}
		//当新节点内的值小于父节点内的值时
		else
		{
			parentcur->left = cur;
		}
		return true;

	}

	Node* Find(const K& key)
	{
		//当根结点为空时
		if (_root == nullptr)
		{
			return nullptr;
		}
		Node* cur = _root;
		Node* parentcur = nullptr;
		while (cur)
		{
			//当key大于当前节点的值
			if (cur->_key < key)
			{
				parentcur = cur;
				cur = cur->right;
			}
			//当key小于当前节点的值
			else if (cur->_key > key)
			{
				parentcur = cur;
				cur = cur->left;
			}
			//当key等于当前节点的值
			else
			{
				return cur;
			}
		}
		//当前二叉树中找不到值为key的节点
		return nullptr;
	}


	bool Erase(const K& key)
	{
		//当根节点为空时
		if (_root == nullptr)
		{
			return false;
		}
		//当前节点
		Node* cur = _root;
		//当前节点的父节点
		Node* parentcur = _root;
		while (cur)
		{
			//当key的值大于当前节点的值
			if (cur->_key < key)
			{
				parentcur = cur;
				cur = cur->right;
			}
			//当key的值大于当前节点的值
			else if (cur->_key > key)
			{
				parentcur = cur;
				cur = cur->left;
			}
			//当key的值等于当前节点的值
			else
			{
				//当要删除的节点左孩子节点为空
				if (cur->left == nullptr)
				{
					//当要删除的节点为根结点时直接改变_root指针的指向,之后再将原根结点释放
					if (cur == _root)
					{
						Node* Right = cur->right;
						delete cur;
						_root = Right;
					}
					else
					{
						//当要删除的节点cur是其父节点的左节点时
						if (cur == parentcur->left)
						{
							parentcur->left = cur->right;
							delete cur;
						}
						//当要删除的节点cur是其父节点的右节点时
						else if (cur == parentcur->right)
						{
							parentcur->right = cur->right;
							delete cur;
						}
					}

					return true;

				}
				//当要删除的节点右孩子节点为空
				else if (cur->right == nullptr)
				{
					//当要删除的节点为根结点时直接改变_root指针的指向,之后再将原根结点释放
					if (cur == _root)
					{
						Node* Left = cur->left;
						delete cur;
						_root = Left;
					}
					else
					{
						//当要删除的节点cur是其父节点的左节点时
						if (cur == parentcur->left)
						{
							parentcur->left = cur->left;
							delete cur;
						}
						//当要删除的节点cur是其父节点的右节点时
						else if (cur == parentcur->right)
						{
							parentcur->right = cur->left;
							delete cur;
						}
					}
					return true;

				}
				//当要删除的节点左右孩子节点都为空
				else
				{
					Node* minrightParent = cur;
					Node* minright = cur->right;
					//找出当前节点右子树中的最左节点
					while (minright->left)
					{
						minrightParent = minright;
						minright = minright->left;
					}
					cur->_key = minright->_key;
					//当最左节点为其父节点的左节点时
					if (minrightParent->left == minright)
					{
						minrightParent->left = minright->right;

					}
					//当最左节点为其父节点的右节点时
					else
					{
						minrightParent->right = minright->right;

					}
					delete minright;
					return true;

				}

			}
		}

		//当找不到要删除的节点就返回false
		return false;

	}

	void Inorder()
	{
		_Inorder(_root);
		cout << endl;
	}




private:
	//头节点
	Node* _root = nullptr;

	void _Inorder(Node* root)
	{
		if (root == nullptr)
		{
			return;
		}
		_Inorder(root->left);
		cout << root->_key << ":" << root->_val << " ";
		_Inorder(root->right);
	}

};


 

在以上我们了解实现的二叉搜索树其实是适用于key的场景,那么接下来就来看看那么场景是属于key的场景

场景1:商场无人值守车库,入⼝进场时扫描⻋牌,记录车牌和入场时间,出口离场时,扫描牌,查找入场时间,用当前时间-⼊场时间计算出停⻋时长,计算出停车费用,缴费后抬杆,车辆离场。

 

场景1:简单中英互译字典,树的结构中(结点)存储key(文)和vlaue(中文),搜索时输⼊英文,则同时查找到了英文对应的中文。
场景3:统计⼀篇文章中单词出现的次数,读取⼀个单词,查找单词是否存在,不存在这个说明第⼀次出现,(单词,1),单词存在,则++单词对应的次数。

以下的两个简单的示例就是使用二叉搜索树的key/val来解决
 

#include"BSTree.h"


int main()
{
	//示例1
//将以下英文单词和对应的中文翻译绑定到一起,当用户输入输入正确的单词就输出其中文意思,否则就输出单词拼写错误
	BSTree<string, string> dict;
	dict.Insert("insert", "插入");
	dict.Insert("erase", "删除");
	dict.Insert("left", "左边");
	dict.Insert("string", "字符串");

	string str;
	while (cin >> str)
	{
		auto ret = dict.Find(str);
		if (ret)
		{
			cout << str << ":" << ret->_val << endl;
		}
		else
		{
			cout << "单词拼写错误" << endl;
		}
	}


    //示例2
//统计字符串数组当中各个水果的出现次数
	string strs[] = { "苹果", "西瓜", "苹果", "樱桃", "苹果", "樱桃", "苹果", "樱桃", "苹果" };
	// 统计水果出现的次
	BSTree<string, int> countTree;
	for (auto str : strs)
	{
		auto ret = countTree.Find(str);
		if (ret == NULL)
		{
			countTree.Insert(str, 1);
		}
		else
		{
			ret->_val++;
		}
	}
	countTree.Inorder();



	return 0;
}

以上就是本篇的区别内容了,希望能得到你的点赞和收藏 ❤️

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2247781.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Leetcode647. 回文子串(HOT100)

链接 代码&#xff1a; class Solution { public:int countSubstrings(string s) {int res 0;for(int i 0;i<s.size();i){for(int j i,k i;j>0&&k<s.size();j--,k){if(s[j]!s[k])break;else res;}for(int j i,k i1;j>0&&k<s.size();j--…

ubuntu, 安装部署comfyui,记录2:下载模型GGuf及测试

0.清除工作流 1.安装manager 2024年最新ComfyUI汉化及manager插件安装详解&#xff01;_comfyui-manager-CSDN博客 ComfyUI Manager安装 转到ComfyUI的安装目录ComfyUI/custom_nodes; 使用git拉取ComfyUI Manager&#xff0c;git clone https://github.com/ltdrdata/Comf…

【Y20030006】基于php+mysql的课程学习网站的设计与实现(附源码 配置 文档)

网络购物商城的设计与实现 1.摘要2.开发目的和意义3.系统功能设计4.系统界面截图5.源码获取 1.摘要 随着互联网的普及和在线教育的兴起&#xff0c;课程学习网站已经成为越来越多人获取知识和提升技能的重要途径。在这样的背景下&#xff0c;开发一个基于Laravel框架的课程学习…

O2O: (BOORL) Bayesian Design Principles for Offline-to-Online RL

ICML 2024 paper code Intro O2O如何避免悲观学习导致sample efficiency较低&#xff0c;亦或者乐观估计导致的performance drop。本文提出贝叶斯准则&#xff0c;指导在线学习过程中的探索和利用。通过构建一个Q值相关的信念分布&#xff0c;agent可以对不同策略的优劣有一个…

Hyper-V配置-cnblog

启用Hyper-V以在 Windows 10上创建虚拟机 &#xff08;1&#xff09;控制面板检查系统要求&#xff1a; 确保您的计算机符合 Hyper-V 的系统要求。通常情况下&#xff0c;您的计算机需要运行 Windows 10 专业版、企业版或教育版&#xff0c;并且具有启用了虚拟化技术的处理器。…

基础免杀 从.rsrc加载shellcode上线

.rsrc 段是PE文件中的一个特定部分&#xff0c;专门用来存储资源数据。这些资源通常包括图标、位图、字符串表、对话框、菜单、版本信息、字体等 具体的shellcode加载方式不在此探讨 在这使用传统的指针执行 WindowsAPI 需要用到如下API FindResource 获取指定资源的信息块…

Excel把其中一张工作表导出成一个新的文件

excel导出一张工作表 一个Excel表里有多个工作表&#xff0c;怎么才能导出一个工作表&#xff0c;让其生成新的Excel文件呢&#xff1f; 第一步&#xff1a;首先打开Excel表格&#xff0c;然后选择要导出的工作表的名字&#xff0c;比如“Sheet1”&#xff0c;把鼠标放到“She…

第四期书生大模型实战营——基础岛第4关-L1G4000-InternLM + LlamaIndex RAG 实践

Tutorial 基础任务 任务要求1&#xff08;必做&#xff0c;参考readme_api.md&#xff09;&#xff1a;基于 LlamaIndex 构建自己的 RAG 知识库&#xff0c;寻找一个问题 A 在使用 LlamaIndex 之前 浦语 API 不会回答&#xff0c;借助 LlamaIndex 后 浦语 API 具备回答 A 的能…

搭建文件服务器并使用Qt实现文件上传和下载(带账号和密码)

文章目录 0 背景1 搭建文件服务器2 代码实现文件上传和下载2.1 在pro文件中添加网络支持2.2 创建网络管理类2.3 文件上传2.4 文件下载 3 扩展&#xff08;其他方法实现文件上传和下载&#xff09;3.1 python3.2 npm3.3 ftp服务器 4 完整的代码 0 背景 因为需要使程序具备在远程…

【初阶数据结构和算法】leetcode刷题之设计循环队列

文章目录 一、实现循环队列1.大致思路分析2.循环队列的结构定义和初始化结构定义初始化 3.循环队列的判空和判满判空和判满难点分析判空判满 4.循环队列的入队列和出队列入队列出队列 5.循环队列取队头和队尾元素取队头元素取队尾元素 6.循环队列的销毁7.最后题解源码 一、实现…

llama-factory 系列教程 (七),Qwen2.5-7B-Instruct 模型微调与vllm部署详细流程实战

文章目录 介绍llama-factory 安装装包下载模型 微调模型数据集训练模型 微调后的模型推理 介绍 时隔已久的 llama-factory 系列教程更新了。本篇文章是第七篇&#xff0c;之前的六篇&#xff0c;大家酌情选看即可。 因为llama-factory进行了更新&#xff0c;我前面几篇文章的实…

矩阵的重复

重复时自身也算一次重复 r e p m a t ( r e p e a t repmat(repeat repmat(repeat m a t l a b ) matlab) matlab)重复矩阵函数 ( ( ( 对矩阵整体 ) ) ) r e p m a t ( a , m , n ) repmat(a,m,n) repmat(a,m,n)将矩阵纵向重复 m m m次&#xff0c;横向重复 n n n次 r e …

【三维重建】windows10环境配置tiny-cuda-nn详细教程

1. 前言 本人在复现water-splatting时&#xff0c;需要配置tiny-cuda-nn&#xff0c;与此同时&#xff0c;出现了很多问题&#xff0c;在此进行简单概述。 2.安装Pytorch 环境版本要求保持一致&#xff1a;CUDA(物理机)&#xff0c;Pytorch&#xff0c;CUDA Toolkit 注意这里…

Js-函数-03

函数定义 在java中我们为了提高代码的复用性&#xff0c;可以使用方法。同样&#xff0c;在JavaScript中可以使用函数来完成相同的事情。JavaScript中的函数被设计为执行特定任务的代码块&#xff0c;通过关键字function来定义。 <!DOCTYPE html> <html lang"en…

MySQL45讲 第29讲 如何判断一个数据库是不是出问题了?——阅读总结

文章目录 MySQL45讲 第二十九讲 如何判断一个数据库是不是出问题了&#xff1f;——阅读总结一、检测数据库实例健康状态的重要性二、常见检测方法及问题分析&#xff08;一&#xff09;select 1 判断法&#xff08;二&#xff09;查表判断法&#xff08;三&#xff09;更新判断…

IO多路复用(Linux epoll)

文章目录 一、IO多路复用介绍1. 缓存 I/O (各种IO模型缘起) 二、目前有哪些IO多路复用的方案三、关联基础知识1. 用户空间和内核空间2. 文件描述符fd 四、Linux IO多路复用 select五、Linux IO多路复用 epoll1. epoll 介绍2. epoll只提供三个函数ET模式与LT模式 3. demo验证 六…

【数据结构与算法】相交链表、环形链表(判断是否有环)、环形链表(返回入环节点)

主页&#xff1a;HABUO&#x1f341;主页&#xff1a;HABUO &#x1f341;如果再也不能见到你&#xff0c;祝你早安&#xff0c;午安&#xff0c;晚安&#x1f341; 1.相交链表 题目&#xff1a;给你两个单链表的头节点 headA 和 headB &#xff0c;请你找出并返回两个单链表…

Move 合约部署踩坑笔记:如何解决 Sui 客户端发布错误Committing lock file

Move 共学活动&#xff1a;快速上手 Move 开发 为了帮助更多开发者快速了解和掌握 Move 编程语言&#xff0c;Move 共学活动由 HOH 社区、HackQuest、OpenBuild、KeyMap 联合发起。该活动旨在为新手小白提供一个良好的学习平台&#xff0c;带领大家一步步熟悉 Move 语言&#…

【C语言】野指针问题详解及防范方法

博客主页&#xff1a; [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: C语言 文章目录 &#x1f4af;前言&#x1f4af;什么是野指针&#xff1f;&#x1f4af;未初始化的指针代码示例问题分析解决方法 &#x1f4af;指针越界访问代码示例问题分析解决方法 &#x1f4af;指向已释放内存的…

关于如何在k8s中搭建一个nsfw黄图鉴定模型

随着现在应用内图片越来越多&#xff0c;安全审查也是必不可少的一个操作了 下面手把手教你如何将huggingface中的黄图检测模型部署到自己的服务器上去 1.找到对应的模型 nsfw_image_detection 2.在本地先验证如何使用 首先安装transformers python库 pip install transform…