【数据结构】二叉搜索树(二叉排序树)

🌟🌟作者主页:ephemerals__

🌟🌟所属专栏:数据结构

目录

前言

一、什么是二叉搜索树

二、二叉搜索树的实现

节点

属性和接口的声明

插入

查找

删除

拷贝构造

析构

中序遍历

三、二叉搜索树的性能分析

总结


前言

        之前,我们学习了树这一数据结构,并尝试实现了堆以及链式二叉树的相关功能:

【数据结构】树型结构详解 + 堆的实现(c语言)(附源码)_树形结构 编译器-CSDN博客

【数据结构】二叉树(c语言)(附源码)_二叉树数据结构代码-CSDN博客

今天,我们将在简单二叉树的基础上,进一步学习一种更为复杂的结构——二叉搜索树

        之前我们利用c语言实现了顺序表、链表、二叉树等数据结构。但是在实现一些复杂数据结构时,c语言显得相对繁琐,并且容易出现代码冗余的问题。由于我们现在已经具备了一定的c++代码基础,因此在之后的数据结构学习中,我们将用c++实现。

正文开始

一、什么是二叉搜索树

        二叉搜索树(Binary Search Tree),也叫二叉排序树或者二叉查找树, 是一种特殊的二叉树结构。它或者是一棵空树,或者具有以下特点:

1. 如果根节点的左子树不为空,则左子树上的所有节点都小于等于根节点的值。

2. 如果根节点的右子树不为空,则右子树上的所有节点都大于等于根节点的值。

3. 它的每一棵子树也都符合前两点特性(每棵子树也都是二叉搜索树)。

简单地说,二叉搜索树是一棵中序遍历结果有序的二叉树。

一般情况下,二叉搜索树不允许有相同的值出现。当然,你在设计二叉搜索树的时候也可以允许相同值出现,只要满足二叉搜索树的特性即可。注意:二叉搜索树不一定是一棵完全二叉树。

二、二叉搜索树的实现

        接下来,我们尝试实现二叉搜索树的结构及其常用方法。

节点

        首先是它的节点的结构。二叉搜索树的节点包含一个数据域和指向两孩子的指针:

//节点
template<class K>
struct BSTNode
{
	K _key;//数据域
	BSTNode<K>* _left;//指向左孩子的指针
	BSTNode<K>* _right;//指向右孩子的指针

	//节点的默认构造
	BSTNode(const K& key)
		:_key(key)
		, _left(nullptr)
		, _right(nullptr)
	{}
};

可以看到,节点的定义当中,模板参数被命名为"K"。这里的"K"通常表示(key)的类型,特别应用于key_value(键值对,可以理解为一个二元组,每一个key都有对应的value)的场景。一般情况下,key的值是不允许修改的,但是其对应的value可以修改。本次实现二叉搜索树的过程当中,我们仅实现带有key的结构,至于key_value结构,会在之后的AVL树和红黑树中实现。

属性和接口的声明

        接下来是二叉搜索树类的成员变量和成员函数的声明:

//二叉搜索树
template<class K>
class BST
{
	typedef BSTNode<K> Node;//重命名,简化代码
public:
	//强制生成无参构造
	BST() = default;

	//拷贝构造
	BST(const BST<K>& t);

	//析构
	~BST();

	//插入
	bool Insert(const K& key);

	//删除
	bool Erase(const K& key);

	//查找
	Node* Find(const K& key);

	//中序遍历
	void Inorder();
private:
	Node* _root = nullptr;//根节点的指针
};

插入

        现在我们开始实现二叉搜索树节点的插入。为了保持搜索树的特性,有如下插入步骤:

1. 如果树为空,则直接将新节点赋值给根节点的指针。

2. 如果树不为空,则需要根据新节点的值进行搜索,如果某节点的值大于新节点,则向右走;若小于新节点,则向左走。走到空位置之后,按照值插入节点即可。

3. 如果要插入重复的值,则遇到相同值的节点之后可以向左走,也可以向右走,直到找到空位置插入。(我们的实现默认不支持插入重复值

代码实现:

//插入
bool Insert(const K& key)
{
	if (_root == nullptr)//树为空,直接插入
	{
		_root = new Node(key);
		return true;
	}
	else//树不为空,进行搜索
	{
		Node* cur = _root;//从根节点开始搜索
		Node* parent = nullptr;//记录cur的父亲节点
		while (cur)//走到空为止
		{
			if (key < cur->_key)//值小向左走
			{
				parent = cur;
				cur = cur->_left;
			}
			else if (key > cur->_key)//值大向右走
			{
				parent = cur;
				cur = cur->_right;
			}
			else//这里不允许重复的值插入
			{
				return false;
			}
		}

		//此时cur走到空,进行插入
		cur = new Node(key);
		if (key < parent->_key)//值小,往左边插
		{
			parent->_left = cur;
		}
		else//值大,往右边插
		{
			parent->_right = cur;
		}
		return true;
	}
}

这里我们定义了一个parent指针,用于记录cur的父亲节点。当cur走到空时,parent刚好指向插入位置的父亲节点,然后与parent值进行大小比较,插入新节点。

查找

        二叉搜索树的特性使其具备了较高的查找效率。查找步骤如下:

1. 从根节点开始,与要查找的值进行比较,如果要找的值比根小,则向左走,否则向右走。循环往复。

2. 如果走到空,则说明不存在,返回空指针。

3. 如果不支持插入重复值,找到后返回节点地址。

4. 如果支持插入重复值,则找到后一般返回中序遍历结果的第一个节点

代码实现:

//查找
Node* Find(const K& key)
{
	Node* cur = _root;//从根节点开始搜索
	while (cur)//走到空为止
	{
		if (key < cur->_key)//值小向左走
		{
			cur = cur->_left;
		}
		else if (key > cur->_key)//值大向右走
		{
			cur = cur->_right;
		}
		else//相等说明找到了,返回节点地址
		{
			return cur;
		}
	}

	//走到空说明没找到,返回空指针
	return nullptr;
}

删除

        删除节点是所有接口中实现难度最大的一个。我们默认需要按值来删除元素,所以要先进行查找,找到之后再删除该节点。删除节点之后,为了保持二叉搜索树的原有特性,就需要调整其他节点。根据被删除的节点,有四种情况需要分析讨论:

1. 被删除节点的左右孩子均为空

2. 被删除的节点只有左孩子

3. 被删除的节点只有右孩子

4.  被删除的节点既有左孩子,又有右孩子

这里说明一下寻找右子树最小值的方法:从右孩子开始,一直访问其左孩子节点,直到访问到空为止。这里访问到的最后一个节点即是右子树的最小值。(左子树最大值也同理)

接下来,我们尝试实现删除节点的操作:

//删除
bool Erase(const K& key)
{
	//首先按值查找要被删除的节点
	Node* cur = _root;
	Node* parent = nullptr;//记录父亲节点
	while (cur)
	{
		if (key < cur->_key)//值小向左走
		{
			parent = cur;
			cur = cur->_left;
		}
		else if (key > cur->_key)//值大向右走
		{
			parent = cur;
			cur = cur->_right;
		}
		else//找到了,开始删除
		{
			//有一个左孩子的情况 和 没有孩子的情况
			if (cur->_right == nullptr)
			{
				//特殊情况,要删除的节点是根节点
				if (cur == _root)
				{
					_root = cur->_left;//直接让根指针指向左孩子
				}
				else
				{
					//先判断自己是父亲节点的哪一个孩子
					//然后将左孩子托付给父亲节点
					if (cur == parent->_left)
						parent->_left = cur->_left;
					else
						parent->_right = cur->_left;
				}
				delete cur;//删除该节点
				return true;
			}
			//有一个右孩子的情况
			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;
				}
				delete cur;//删除该节点
				return true;
			}
			//有两个孩子的情况
			else
			{
				//首先寻找右子树的最小值
				Node* rightMin = cur->_right;//从右孩子开始搜索
				Node* rightMinParent = cur;//记录最小值的父亲节点

				while (rightMin->_left)//left走到空,rightMin即为右子树的最小值
				{
					rightMinParent = rightMin;
					rightMin = rightMin->_left;//一直向左走,找最小值
				}

				//将最小值赋值给被删除节点
				cur->_key = rightMin->_key;

				//将替代节点的右孩子托付给父亲
				//先判断自己是父亲节点的哪一个孩子
				if (rightMin == rightMinParent->_left)
					rightMinParent->_left = rightMin->_right;
				else
					rightMinParent->_right = rightMin->_right;
				delete rightMin;//删除该节点
				return true;
			}
		}
	}
	//没找到,返回false
	return false;
}

这里有两点需要注意:

1. 第一种情况可以和第二/第三种情况合并,将空指针托付给父亲节点不会影响整体操作。

2. 第四种情况当中,如果我们找的是右子树的最小值,那么它或是叶子节点(第一种情况),或只有右孩子(第三种情况),不会有左孩子;如果我们找的是左子树的最大值,那么它或是叶子节点(第一种情况),或只有左孩子(第二种情况),不会有右孩子。所以代码中可以非常确定托付哪一个孩子

拷贝构造

        接下来我们实现拷贝构造。该接口比较简单,直接递归拷贝即可。代码如下:

//拷贝构造
BST(const BST<K>& t)
{
	_root = _Copy(t._root);
}

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;//返回根节点
}

析构

        与拷贝构造相同,析构时递归释放每一个节点。代码如下:

//析构
~BST()
{
	_Destroy(_root);
}
void _Destroy(Node* root)
{
	if (root == nullptr)
	{
		return;
	}
	_Destroy(root->_left);//销毁左子树
	_Destroy(root->_right);//销毁右子树
	delete root;//删除根节点
}

中序遍历

        由于二叉搜索树的中序遍历结果是有序的,所以我们来实现一个中序遍历接口。中序遍历的代码与传统的二叉树完全相同。

实现如下:

//中序遍历
void Inorder()
{
	_Inorder(_root);
	cout << endl;
}
void _Inorder(Node* root)
{
	if (root == nullptr)
	{
		return;
	}
	_Inorder(root->_left);//遍历左子树
	cout << root->_key << ' ';//访问根节点
	_Inorder(root->_right);//遍历右子树
}

最后,我们来使用一下我们实现的二叉搜索树:

int main()
{
	BST<int> t;
	int a[] = { 8, 3, 1, 10, 6, 4, 7, 14, 13 };
	for (auto& e : a)//循环插入
	{
		t.Insert(e);
		t.Inorder();
	}
	cout << endl;
	for (auto& e : a)//循环删除
	{
		t.Erase(e);
		t.Inorder();
	}
	return 0;
}

 运行结果:

程序全部代码

二叉搜索树的全部实现代码如下:

#include <iostream>
using namespace std;

//节点
template<class K>
struct BSTNode
{
	K _key;//数据域
	BSTNode<K>* _left;//指向左孩子的指针
	BSTNode<K>* _right;//指向右孩子的指针

	//节点的默认构造
	BSTNode(const K& key)
		:_key(key)
		, _left(nullptr)
		, _right(nullptr)
	{}
};

//二叉搜索树
template<class K>
class BST
{
	typedef BSTNode<K> Node;//重命名,简化代码
public:
	//强制生成无参构造
	BST() = default;

	//拷贝构造
	BST(const BST<K>& t)
	{
		_root = _Copy(t._root);
	}

	//析构
	~BST()
	{
		_Destroy(_root);
	}

	//插入
	bool Insert(const K& key)
	{
		if (_root == nullptr)//树为空,直接插入
		{
			_root = new Node(key);
			return true;
		}
		else//树不为空,进行搜索
		{
			Node* cur = _root;//从根节点开始搜索
			Node* parent = nullptr;//记录cur的父亲节点
			while (cur)//走到空为止
			{
				if (key < cur->_key)//值小向左走
				{
					parent = cur;
					cur = cur->_left;
				}
				else if (key > cur->_key)//值大向右走
				{
					parent = cur;
					cur = cur->_right;
				}
				else//这里不允许重复的值插入
				{
					return false;
				}
			}

			//此时cur走到空,进行插入
			cur = new Node(key);
			if (key < parent->_key)//值小,往左边插
			{
				parent->_left = cur;
			}
			else//值大,往右边插
			{
				parent->_right = cur;
			}
			return true;
		}
	}

	//删除
	bool Erase(const K& key)
	{
		//首先按值查找要被删除的节点
		Node* cur = _root;
		Node* parent = nullptr;//记录父亲节点
		while (cur)
		{
			if (key < cur->_key)//值小向左走
			{
				parent = cur;
				cur = cur->_left;
			}
			else if (key > cur->_key)//值大向右走
			{
				parent = cur;
				cur = cur->_right;
			}
			else//找到了,开始删除
			{
				//有一个左孩子的情况 和 没有孩子的情况
				if (cur->_right == nullptr)
				{
					//特殊情况,要删除的节点是根节点
					if (cur == _root)
					{
						_root = cur->_left;//直接让根指针指向左孩子
					}
					else
					{
						//先判断自己是父亲节点的哪一个孩子
						//然后将左孩子托付给父亲节点
						if (cur == parent->_left)
							parent->_left = cur->_left;
						else
							parent->_right = cur->_left;
					}
					delete cur;//删除该节点
					return true;
				}
				//有一个右孩子的情况
				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;
					}
					delete cur;//删除该节点
					return true;
				}
				//有两个孩子的情况
				else
				{
					//首先寻找右子树的最小值
					Node* rightMin = cur->_right;//从右孩子开始搜索
					Node* rightMinParent = cur;//记录最小值的父亲节点

					while (rightMin->_left)//left走到空,rightMin即为右子树的最小值
					{
						rightMinParent = rightMin;
						rightMin = rightMin->_left;//一直向左走,找最小值
					}

					//将最小值赋值给被删除节点
					cur->_key = rightMin->_key;

					//将替代节点的右孩子托付给父亲
					//先判断自己是父亲节点的哪一个孩子
					if (rightMin == rightMinParent->_left)
						rightMinParent->_left = rightMin->_right;
					else
						rightMinParent->_right = rightMin->_right;
					delete rightMin;//删除该节点
					return true;
				}
			}
		}
		//没找到,返回false
		return false;
	}

	//查找
	Node* Find(const K& key)
	{
		Node* cur = _root;//从根节点开始搜索
		while (cur)//走到空为止
		{
			if (key < cur->_key)//值小向左走
			{
				cur = cur->_left;
			}
			else if (key > cur->_key)//值大向右走
			{
				cur = cur->_right;
			}
			else//相等说明找到了,返回节点地址
			{
				return cur;
			}
		}

		//走到空说明没找到,返回空指针
		return nullptr;
	}

	//中序遍历
	void Inorder()
	{
		_Inorder(_root);
		cout << endl;
	}
private:
	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;//删除根节点
	}

	void _Inorder(Node* root)
	{
		if (root == nullptr)
		{
			return;
		}
		_Inorder(root->_left);//遍历左子树
		cout << root->_key << ' ';//访问根节点
		_Inorder(root->_right);//遍历右子树
	}

	Node* _root = nullptr;//根节点的指针
};

三、二叉搜索树的性能分析

        接下来,我们简单分析一下二叉搜索树的性能。 

设二叉树的节点数为N

在较好情况下,二叉搜索树接近于完全二叉树的状态,它的高度为log₂N

在极端情况下,二叉搜索树是单支树的状态,高度为N

综合而言, 二叉搜索树增、删、改的时间复杂度为O(N)

        在较好情况下,其增删改的效率可接近于O(logN),但是它的效率受高度影响较大。所以传统的二叉搜索树显然不满足我们高效查找的需求。之后我们会学习自平衡的二叉搜索树(如AVL树、红黑树),它们会尽量降低二叉搜索树的高度,提高效率。

总结

        今天我们学习了一种特殊的二叉树结构--二叉搜索树,它的特性使其具备了较好的查找效率。掌握二叉搜索树的知识,将为我们后续学习STL中的setmap容器打下坚实的基础。如果你觉得博主讲的还不错,就请留下一个小小的赞在走哦,感谢大家的支持❤❤❤

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

如何启用本机GPU硬件加速猿大师播放器网页同时播放多路RTSP H.265 1080P高清摄像头RTSP视频流?

目前市面上主流播放RTSP视频流的方式是用服务器转码方案&#xff0c;这种方案的好处是兼容性更强&#xff0c;可以用于不同的平台&#xff0c;比如&#xff1a;Windows、Linux或者手机端&#xff0c;但是缺点也很明显&#xff1a;延迟高、播放高清或者同时播放多路视频视频容易…

乘积求导法则、除法求导法则和链式求导法则

乘积求导法则、除法求导法则和链式求导法则 1. Constant multiples of functions (函数的常数倍)2. Sums and differences of functions (函数和与函数差)3. Products of functions via the product rule (通过乘积法则求积函数的导数)4. Quotients of functions via the quoti…

2个GitHub上最近比较火的Java开源项目

1. SpringBlade 微服务架构 标题 SpringBlade 微服务架构 摘要 SpringBlade 是一个由商业级项目升级优化而来的微服务架构&#xff0c;采用Spring Boot 3.2、Spring Cloud 2023等核心技术构建&#xff0c;遵循阿里巴巴编码规范&#xff0c;提供基于React和Vue的两个前端框架&am…

Ubuntu 安装 MariaDB

安装 MariaDB具体步骤 1、更新软件包索引&#xff1a; sudo apt update2、安装 MariaDB 服务器&#xff1a; sudo apt install mariadb-server3、启动 MariaDB 服务&#xff08;如果未自动启动&#xff09;&#xff1a; sudo systemctl start mariadb4、设置 MariaDB 开机启…

一体化数据安全平台uDSP 入选【年度创新安全产品 TOP10】榜单

近日&#xff0c;由 FreeBuf 主办的 FCIS 2024 网络安全创新大会在上海隆重举行。大会现场揭晓了第十届 WitAwards 中国网络安全行业年度评选获奖名单&#xff0c;该评选自 2015 年举办以来一直饱受赞誉&#xff0c;备受关注&#xff0c;评选旨在以最专业的角度和最公正的态度&…

相同的二叉树

给你两棵二叉树的根节点 p 和 q &#xff0c;编写一个函数来检验这两棵树是否相同。 如果两个树在结构上相同&#xff0c;并且节点具有相同的值&#xff0c;则认为它们是相同的。 示例 1&#xff1a; 输入&#xff1a;p [1,2,3], q [1,2,3] 输出&#xff1a;true示例 2&…

百度地图JSAPI WebGL v1.0类参考

百度地图JSAPI WebGL v1.0类参考 核心类 Map 此类是地图API的核心类&#xff0c;用来实例化一个地图。请注意WebGL版本的地图API的命名空间是BMapGL。 示例&#xff1a;const map new BMapGL.Map(container); 构造函数描述Map(container: String | HTMLElement, opts: Map…

【k8s】监控metrics-server

metrics-server介绍 Metrics Server是一个集群范围的资源使用情况的数据聚合器。作为一个应用部署在集群中。Metric server从每个节点上KubeletAPI收集指标&#xff0c;通过Kubernetes聚合器注册在Master APIServer中。为集群提供Node、Pods资源利用率指标。 就像Linux 系统一样…

AI 声音:数字音频、语音识别、TTS 简介与使用示例

在现代 AI 技术的推动下&#xff0c;声音处理领域取得了巨大进展。从语音识别&#xff08;ASR&#xff09;到文本转语音&#xff08;TTS&#xff09;&#xff0c;再到个性化声音克隆&#xff0c;这些技术已经深入到我们的日常生活中&#xff1a;语音助手、自动字幕生成、语音导…

ARM CCA机密计算安全模型之硬件强制安全

安全之安全(security)博客目录导读 [要求 R0004] Arm 强烈建议所有 CCA 实现都使用硬件强制的安全(CCA HES)。本文件其余部分假设系统启用了 CCA HES。 CCA HES 是一个可信子系统的租户——一个 CCA HES 主机(Host),见下图所示。它将以下监控安全域服务从应用处理元件(P…

matlab显示sin二维图

1&#xff0c;新建脚本 2、保存脚本 3、脚本命令&#xff1a;clc 清除 脚本命令的信息 clrear all 清除全部 4工作区内容&#xff1a;变量啥的 x0:0.001:2*pi%% 开始 精度 中值 ysin(x) y1cos(x) figure%%产生一个屏幕 plot(x,y)%%打印坐标 title(ysin(x))%%标题 xlabel(…

一万台服务器用saltstack还是ansible?

一万台服务器用saltstack还是ansible? 选择使用 SaltStack 还是 Ansible 来管理一万台服务器&#xff0c;取决于几个关键因素&#xff0c;如性能、扩展性、易用性、配置管理需求和团队的熟悉度。以下是两者的对比分析&#xff0c;帮助你做出决策&#xff1a; SaltStack&…

Flutter 1.2:flutter配置gradle环境

1、在android的模块中进行gradle环境配置 ①在 gradle-wrapper.properties文件中将url配置为阿里云镜像&#xff0c;因为gradle的服务器在国外&#xff0c;国内下载非常慢&#xff0c;也可在官网进行下载 gradle版本下载 gradle版本匹配 阿里云镜像gradle下载 可以通过复制链…

vue 2 父组件根据注册事件,控制相关按钮显隐

目标效果 我不注册事件&#xff0c;那么就不显示相关的按钮 注册了事件&#xff0c;才会显示相关内容 实现思路 组件在 mounted 的时候可以拿到父组件注册监听的方法 拿到这个就可以做事情了 mounted() {console.log(this.$listeners, this.$listeners);this.show.search !…

Vue+Elementui el-tree树只能选择子节点并且支持检索

效果&#xff1a; 只能选择子节点 添加配置添加检索代码 源码&#xff1a; <template><div><el-button size"small" type"primary" clearable :disabled"disabled" click"showSign">危险点评估</el-button>…

PhPMyadmin-漏洞复现

前情条件是&#xff1a;首先将我们的PHP版本设置在5.5以上 注&#xff1a;禁止用于未授权的测试! 首先搭建环境&#xff0c;登录后台 点击》》SQL 查看当前的日志状态 SHOW VARIABLES LIKE general%;因为之前我原来做过所以general_log 是开启的&#xff0c;如果vlau 是OF…

【2024】使用Docker搭建redis sentinel哨兵模式集群全流程(包含部署、测试、错误点指正以及直接部署)

目录&#x1f4bb; 前言**Docker Compose介绍**最终实现效果 一、搭建集群1、创建文件结构2、创建redis节点3、验证节点4、创建sentinel哨兵5、验证Sentinel功能 二、spring连接1、添加依赖2、添加配置3、启动测试 三、直接部署流程1、拉取配置2、修改端口创建 前言 本篇文章主…

泷羽sec-shell编程(完结) 学习笔记

声明&#xff01; 学习视频来自B站up主 **泷羽sec** 有兴趣的师傅可以关注一下&#xff0c;如涉及侵权马上删除文章&#xff0c;笔记只是方便各位师傅的学习和探讨&#xff0c;文章所提到的网站以及内容&#xff0c;只做学习交流&#xff0c;其他均与本人以及泷羽sec团队无关&a…

眼部按摩仪WT2605音频蓝牙语音芯片方案 单芯片实现语音提示及控制/手机无线音频传输功能

随着科技的快速发展&#xff0c;人们的生活方式也在不断改变&#xff0c;智能化、便捷化的产品逐渐成为市场的主流。眼部按摩仪作为一种结合了现代科技与健康生活理念的产品&#xff0c;受到了广大消费者的青睐。而在众多眼部按摩仪中&#xff0c;采用WT2605音频蓝牙芯片的方案…

相交链表和环形链表

&#xff08;一&#xff09;相交链表 相交链表 思路&#xff1a;先分别计算出A列表和B列表的长度&#xff0c;判断它们的尾节点是否相等&#xff0c;如果不相等就不相交&#xff0c;直接返回空。然后让两个列表中的长的列表先走它们的差距步&#xff0c;然后再一起走&#xff…