【数据结构】二叉树算法讲解(定义+算法原理+源码)

news2024/11/23 10:31:15

博主介绍:✌全网粉丝喜爱+、前后端领域优质创作者、本质互联网精神、坚持优质作品共享、掘金/腾讯云/阿里云等平台优质作者、擅长前后端项目开发和毕业项目实战✌有需要可以联系作者我哦!

🍅附上相关C语言版源码讲解🍅

👇🏻 精彩专栏推荐订阅👇🏻 不然下次找不到哟

目录

一、二叉树定义(特点+结构)

二叉树算法性质:

二、算法实现(完整代码)

三、算法总结

二叉树的优点:

 二叉树的缺点:

二叉树的应用:

小结

大家点赞、收藏、关注、评论啦 !

谢谢哦!如果不懂,欢迎大家下方讨论学习哦。

一、二叉树定义(特点+结构)

二叉树是一种树形结构,每个节点最多有两个子节点,分别称为左子节点和右子节点。二叉树具有以下定义和特点:

1. 节点:二叉树是由节点构成的集合。每个节点包含三个基本信息:
   - 数据元素(或称为节点值)。
   - 指向左子节点的指针/引用。
   - 指向右子节点的指针/引用。

2. 根节点: 二叉树中的一个节点被称为根节点,它是整个树的起始节点。一棵二叉树只有一个根节点。

3. 叶子节点:没有子节点的节点被称为叶子节点(或叶节点)。

4. 父节点和子节点: 每个节点都有一个父节点,除了根节点。父节点指向它的子节点。

5. 深度:一个节点的深度是从根节点到该节点的唯一路径的边的数量。根节点的深度为0。

6. 高度/深度: 一棵二叉树的高度(或深度)是树中任意节点的最大深度。

7. 子树:二叉树中的任意节点和它的所有子孙节点组成的集合被称为子树。

8. 二叉搜索树(BST):在二叉搜索树中,每个节点的左子树中的节点值都小于该节点的值,而右子树中的节点值都大于该节点的值。

9. 满二叉树:如果一棵深度为k,且有2^k - 1个节点的二叉树被称为满二叉树。

10. 完全二叉树:对于一棵深度为k的二叉树,除了最后一层外,其它各层的节点数都达到最大值,且最后一层的节点都集中在左边,被称为完全二叉树。

二叉树的定义为:

struct TreeNode {
    int val;                 // 节点值
    TreeNode *left;          // 左子节点指针
    TreeNode *right;         // 右子节点指针
    TreeNode(int x) : val(x), left(NULL), right(NULL) {}
};

上述定义为C++中使用类实现的二叉树节点定义,包含节点值、左子节点指针和右子节点指针。

二叉树算法性质:

你提到的这些性质描述了二叉搜索树(Binary Search Tree,BST)的一些重要特征。让我们逐一解释这些性质:

1. 将任何一个点看作Root节点,则这个点的左子树也是 Binary Search Tree:这表示二叉搜索树中每个节点的左子树都满足二叉搜索树的性质,即左子树上的节点值小于当前节点的值。

2. 将任何一个点看作Root节点,则这个点的右子树也是 Binary Search Tree:类似地,这表明每个节点的右子树都是一个二叉搜索树,右子树上的节点值大于当前节点的值。

3. Binary Search Tree中的最小节点,一定是整棵树中最左下的叶子节点:这是因为最小节点不会有左子节点,只能一直沿着左子树往下走,直到叶子节点。

4. Binary Search Tree中的最大节点,一定是整棵树中最右下的叶子节点:同样,最大节点不会有右子节点,只能一直沿着右子树往下走,直到叶子节点。

这些性质是二叉搜索树在节点排列和结构上的特点,它们使得在二叉搜索树上执行搜索、插入和删除等操作更加高效。通过遵循这些性质,可以确保在整个树结构中维持有序性,使得二叉搜索树成为一种常用的数据结构。

二、算法实现(完整代码)

通过二叉树实现A、B、C、D的简单应用

#include<iostream>
using namespace std;
typedef char DataType;
struct BiNode
{
	DataType data;
	BiNode *lchild,*rchild;
};
//(1)假设二叉树采用链接存储方式存储,分别编写一个二叉树先序遍历的递归
//算法和非递归算法。
class BiTree
{
	public:
		BiTree(){root=Create(root);}//构造函数,建立一颗二叉树
		~BiTree(){Release(root);}//析构函数,释放各个节点的存储空间
		void Preorder(){Preorder(root);}//前序遍历二叉树
		void Inorder(){Inorder(root);}//中序遍历二叉树
		void Postorder(){Postorder(root);}//后序遍历二叉树
		void Levelorder(){Levelorder(root);};//层序遍历二叉树
	private:
		BiNode *root;//指向根节点的头指针
		BiNode *Create(BiNode *bt);//构造函数调用
		void Release(BiNode *bt);//析构函数调用
		void Preorder(BiNode *bt);//前序遍历函数调用
		void Inorder(BiNode *bt);//中序遍历函数调用
		void Postorder(BiNode *bt);//后序遍历函数调用
		void Levelorder(BiNode *bt);//层序遍历函数调用
};
//前序遍历
void BiTree::Preorder(BiNode *bt)
{
	if(bt==NULL)return;//递归调用的结束条件
	else{
		cout<<bt->data<<" ";//访问根节点bt的数据域
		Preorder(bt->lchild);//前序递归遍历bt的左子树
		Preorder(bt->rchild);//前序递归遍历bt的右子树
	}
}
//中序遍历
void BiTree::Inorder(BiNode *bt)
{
	if(bt==NULL)return;//递归调用的结束条件
	else{
		Inorder(bt->lchild);//中序递归遍历bt的左子树
		cout<<bt->data<<" ";//访问根节点的数据域
		Inorder(bt->rchild);//中序递归遍历bt的右子树
	}
}
//后序遍历
void BiTree::Postorder(BiNode *bt)
{
	if(bt==NULL)return;//递归调用的结束条件
	else{
		Postorder(bt->lchild);//后序递归遍历bt的左子树
		Postorder(bt->rchild);//后序递归遍历bt的右子树
		cout<<bt->data<<" ";//访问根节点bt的数据域
	}
}
//层序遍历
void BiTree::Levelorder(BiNode *bt){
	BiNode *Q[100],*q=NULL;
	int front=-1,rear=-1;//队列初始化 
	if(root == NULL) return;//二叉树为空,算法结束
	Q[++rear]=root;//根指针入队
	while(front!=rear){//当队列非空时 
		q=Q[++front];//出队
		cout<<q->data<<" ";
		if(q->lchild!=NULL) Q[++rear]=q->lchild;
		if(q->rchild!=NULL) Q[++rear]=q->rchild; 
	} 
}
//创建二叉树 
BiNode *BiTree::Create(BiNode *bt)
{
	static int i=0;
	char ch;
	string str="AB#D##C##";
	ch=str[i++];
	if(ch=='#')bt=NULL;//建立一棵空树 
	else {
		bt=new BiNode;bt->data=ch;//生成一个结点,数据域为ch
		bt->lchild=Create(bt->lchild);//递归建立左子树
		bt->rchild=Create(bt->rchild);//递归建立右子树
	}
	return bt;
}
//销毁二叉树 
void BiTree::Release(BiNode *bt)
{
	if(bt!=NULL){
		Release(bt->lchild);
		Release(bt->rchild);
		delete bt;
	}
}
int main()
{
	cout<<"创建一棵二叉树"<<endl;
	BiTree T;//创建一颗二叉树
	cout<<"---层序遍历---"<<endl;//A B C D 
	T.Levelorder();
	cout<<endl;
	cout<<"---前序遍历---"<<endl;//A B D C
	T.Preorder();
	cout<<endl;
	cout<<"---中序遍历---"<<endl;//B D A C
	T.Inorder();
	cout<<endl;
	cout<<"---后序遍历---"<<endl;//D B C A
	T.Postorder();
	cout<<endl;
	return 0;
}

执行结果:

序存储的完全二叉树递归先序遍历算法描述(C++)如下:

//完全二叉树的顺序存储结构
#include <iostream>
#include <string.h>
#define MaxSize 100
using namespace std;
typedef char DataType;
class Tree{
	public:
		Tree(string str);//构造函数
		void createTree();//创建二叉树 
		void seqPreorder(int i);//先序遍历二叉树 
		void seqInorder(int i);//中序遍历二叉树 
		void seqPostorder(int i);//后序遍历二叉树 
	private: 
		DataType node[MaxSize];//结点中的数据元素
		int num=0;//二叉树结点个数 
		string str;
};
 
Tree::Tree(string str){
	this->str = str;
} 
 
void Tree::createTree(){
	for(int i = 1;i < str.length()+1 ;i++){
		node[i]=str[i-1];
		num++;
	}
	node[0] = (char)num;
}
 
//顺序存储的完全二叉树递归先序遍历算法描述(C++)如下:
void Tree::seqPreorder(int i){
	if(i==0)//递归调用的结束条件
		return;
	else{
		cout<<"  "<<(char)node[i];//输出根结点
		if(2*i<=(char)node[0])
			seqPreorder(2*i);//先序遍历i的左子树
		else
			seqPreorder(0);
		if(2*i+1<=(char)node[0])
			seqPreorder(2*i+1);//先序遍历i的右子树
		else
			seqPreorder(0); 
	} 
} 
 
//顺序存储的完全二叉树递归中序遍历算法描述(C++)如下:
void Tree::seqInorder(int i){
	if(i==0)//递归调用的结束条件
		return;
	else{
		if(2*i<=(char)node[0])
			seqInorder(2*i);//中序遍历i的左子树
		else
			seqInorder(0);
		cout<<"  "<<(char)node[i];//输出根结点
		if(2*i+1<=(char)node[0])
			seqInorder(2*i+1);//中序遍历i的右子树
		else
			seqInorder(0); 
	} 
} 
 
//顺序存储的完全二叉树递归后序遍历算法描述(C++)如下:
void Tree::seqPostorder(int i){
	if(i==0)//递归调用的结束条件
		return;
	else{
		if(2*i<=(char)node[0])
			seqPostorder(2*i);//后序遍历i的左子树
		else
			seqPostorder(0);
		if(2*i+1<=(char)node[0])
			seqPostorder(2*i+1);//后序遍历i的右子树
		else
			seqPostorder(0); 
		cout<<"  "<<(char)node[i];//输出根结点
	} 
} 
 
// (2)一棵完全二叉树以顺序方式存储,设计一个递归算法,对该完全二叉树进
//行中序遍历。
int main(){
	string str = "ABCDEFGHIJ";
	Tree T(str);//定义对象变量bus
	cout<<"按层序编号的顺序存储所有结点:"<<str<<endl;
	T.createTree();
	cout<<"顺序存储的完全二叉树递归前序递归遍历:"<<endl; 
	T.seqPreorder(1);
	cout<<endl; 
	cout<<"顺序存储的完全二叉树递归中序递归遍历:"<<endl; 
	T.seqInorder(1);
	cout<<endl; 
	cout<<"顺序存储的完全二叉树递归后序递归遍历:"<<endl; 
	T.seqPostorder(1);
	cout<<endl; 
	return 0;
}

三、算法总结

二叉树的优点:

1. 快速查找: 在二叉搜索树(BST)中,查找某个元素的时间复杂度是O(log n),这使得二叉树在查找操作上非常高效。

2. 有序性:BST保持元素的有序性,对于某些应用场景,如快速查找最小值、最大值或在某一范围内的值,二叉树非常有用。

3. 容易插入和删除:在BST中,插入和删除操作相对容易,不需要像其他数据结构一样频繁地移动元素。

4. 中序遍历:通过中序遍历二叉搜索树,可以得到有序的元素序列,这对于某些应用(如构建有序列表)很方便。

 二叉树的缺点:

1. 平衡性:如果不平衡,二叉搜索树的性能可能下降为线性级别,而不再是对数级别。因此,需要采取额外的措施来保持树的平衡,如 AVL 树或红黑树。

2. 对数据分布敏感: 对于某些特定的数据分布,比如按顺序插入的数据,可能导致二叉搜索树退化成链表,性能下降。

二叉树的应用:

1. 数据库索引:在数据库中,二叉搜索树被广泛应用于构建索引结构,以加速数据的检索。

2. 表达式解析:二叉树可用于构建表达式树,用于解析和求解数学表达式。

3. 哈夫曼编码:二叉树用于构建哈夫曼树,实现有效的数据压缩算法。

4. 文件系统:在文件系统的目录结构中,可以使用二叉树来组织和管理文件。

5. 网络路由:用于构建路由表,支持快速而有效的网络数据包路由。

6. 编译器设计: 语法分析阶段通常使用二叉树来构建语法树,以便后续的编译步骤。

7. 游戏开发:在游戏开发中,二叉树可以用于实现场景图、动画系统等。

8. 排序算法:一些排序算法,如快速排序,就是通过构建和操作二叉树来实现的。

总体而言,二叉树在计算机科学领域的应用非常广泛,它的特性使得它适用于多种数据管理和搜索场景。在实际应用中,需要根据具体情况选择合适的二叉树变体以及适当的平衡策略。

大家点赞、收藏、关注、评论啦 !

谢谢哦!如果不懂,欢迎大家下方讨论学习哦。

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

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

相关文章

【蓝桥杯--图论】Dijkstra、Ballman-Ford、Spfa、Floyd

今日语录&#xff1a;每一次挑战都是一次成长的机会 文章目录 朴素DIjkstra堆优化的DijkstraBallman-FordFloydSpfa(求最短路)Spfa&#xff08;求是否含有负权&#xff09; 如上所示即为做题时应对的方法 朴素DIjkstra 引用与稠密图&#xff0c;即m<n^2 #include<iostrea…

Java项目:基于ssm框架实现的电影评论系统(ssm+B/S架构+源码+数据库+毕业论文)

一、项目简介 本项目是一套ssm826基于ssm框架实现的电影评论系统&#xff0c;主要针对计算机相关专业的正在做毕设的学生与需要项目实战练习的Java学习者。 包含&#xff1a;项目源码、数据库脚本等&#xff0c;该项目附带全部源码可作为毕设使用。 项目都经过严格调试&#x…

LINUX基础培训十四之系统参数介绍

前言、本章学习目标 了解LINUX中常见系统内核参数掌握常见系统参数优化方法 一、系统参数简介 Linux内核有非常多的参数&#xff0c;而对这些内核参数的修改会尽可能的提高内核的稳定性&#xff0c;并且&#xff0c;在业务高峰期的时候&#xff0c;可以保证内核尽可能的稳定…

Elasticsearch:2023 年 Lucene 领域发生了什么?

作者&#xff1a;来自 Elastic Adrien Grand 2023 年刚刚结束&#xff0c;又是 Apache Lucene 开发活跃的一年。 让我们花点时间回顾一下去年的亮点。 社区 2023 年&#xff0c;有&#xff1a; 5 个次要版本&#xff08;9.5、9.6、9.7、9.8 和 9.9&#xff09;&#xff0c;1 …

力扣740. 删除并获得点数

动态规划 思路&#xff1a; 选择元素 x&#xff0c;获得其点数&#xff0c;删除 x 1 和 x - 1&#xff0c;则其他的 x 的点数也会被获得&#xff1b;可以将数组转换成一个有序 map&#xff0c;key 为 x&#xff0c; value 为对应所有 x 的和&#xff1b;则问题转换成了不能同…

uniapp中打包Andiord app,在真机调试时地图以及定位功能可以正常使用,打包成app后失效问题(高德地图)

踩坑uniapp中打包Andiord app&#xff0c;在真机调试时地图以及定位功能可以正常使用&#xff0c;打包成app后失效问题_uniapp真机调试高德地图正常 打包apk高德地图就不加载-CSDN博客 问题&#xff1a; 目前两个项目&#xff0c;一个项目是从另一个项目里面分割出来的一整套…

对MODNet 主干网络 MobileNetV2的剪枝探索

目录 1 引言 1.1 MODNet 原理 1.2 MODNet 模型分析 2 MobileNetV2 剪枝 2.1 剪枝过程 2.2 剪枝结果 2.2.1 网络结构 2.2.2 推理时延 2.3 实验结论 3 模型嵌入 3.1 模型保存与加载 法一&#xff1a;保存整个模型 法二&#xff1a;仅保存模型的参数 小试牛刀 小结…

【Android Gradle 插件】Gradle 基础配置 ④ ( Gradle Wrapper 配置作用 | Gradle 下载的依赖库存放位置 )

一、Gradle Wrapper 配置作用 gradle wrapperdistributionBaseGRADLE_USER_HOME distributionPathwrapper/dists distributionUrlhttps\://services.gradle.org/distributions/gradle-6.7.1-bin.zip zipStoreBaseGRADLE_USER_HOME zipStorePathwrapper/distsGradle Wrapper 配…

〔保姆级教学〕2024华数杯国际赛B题完整参考论文22页+四小问matlab代码+数据集+可视化高清图表

光伏发电 一、问题分析&#xff08;完整资料在文末&#xff09; 问题一&#xff1a; 首先题目要求得到电能行业与经济状况、居民消费水平、城市化率和市场化程度等因素的关系&#xff0c;并对供电量进行预测。其中&#xff0c;电能采用供电量数据、经济采用GDP数据&#xff…

C++(Qt)软件调试---静态分析工具clang-tidy(18)

C(Qt)软件调试—静态分析工具clang-tidy&#xff08;18&#xff09; 文章目录 C(Qt)软件调试---静态分析工具clang-tidy&#xff08;18&#xff09;1、概述2、clang-tidy基本用法3、目前已有检查项4、Qt Creator中安装clang-tidy5、Qt Creator中使用clang-tidy6、Clang-Tidy配置…

摄影后期照片编辑 -- Lightroom Classic 2024

Lightroom Classic 2024是一款强大的数字照片管理和编辑软件&#xff0c;适合摄影师、图片编辑或者任何需要处理大量照片的人使用。 首先&#xff0c;Lightroom Classic 2024具有直观且易于使用的界面&#xff0c;使得用户可以快速浏览、组织和编辑照片。它支持各种RAW格式的照…

【网络】传输层TCP协议

目录 一、概述 2.1 运输层的作用引出 2.2 传输控制协议TCP 简介 2.3 TCP最主要的特点 2.4 TCP连接 二、TCP报文段的首部格式 三、TCP的运输连接管理 3.1 TCP的连接建立(三次握手) 3.2 为什么是三次握手&#xff1f; 3.3 为何两次握手不可以呢&#xff1f; 3.4 TCP的…

附1:k8s服务器初始化

转载说明&#xff1a;如果您喜欢这篇文章并打算转载它&#xff0c;请私信作者取得授权。感谢您喜爱本文&#xff0c;请文明转载&#xff0c;谢谢。 关联文章&#xff1a; 《RKE快速搭建离线k8s集群并用rancher管理界面》 1.创建普通用户sre并赋予sudo权限 # adduser sre # ec…

vue封装接口

目录 封装接口前缀 配置逻辑 接口存放文件 配置代理 获取数据方法 封装接口前缀 config.js const serverConfig {baseURL: "https://xxx.xxxxxxxx.com/api", // 请求基础地址,可根据环境自定义useTokenAuthorization: false, // 是否开启 token 认证};export …

JoyRL Actor-Critic算法

策略梯度算法的缺点 这里策略梯度算法特指蒙特卡洛策略梯度算法&#xff0c;即 REINFORCE 算法。 相比于 DQN 之类的基于价值的算法&#xff0c;策略梯度算法有以下优点。 适配连续动作空间。在将策略函数设计的时候我们已经展开过&#xff0c;这里不再赘述。适配随机策略。由…

你必须了解的羊奶知识,一文悉数为你揭晓

你必须了解的羊奶知识&#xff0c;一文悉数为你揭晓 羊奶&#xff0c;作为一种营养丰富的乳制品&#xff0c;近年来备受关注。许多人选择羊奶作为替代牛奶的选择&#xff0c;因为它被认为更易消化&#xff0c;并且具有许多健康益处。在本文中&#xff0c;小编羊大师将为大家介…

Mac 上网易云音乐 ncm 格式文件如何转换为 mp3 音频文件?嗨格式转换器

hello朋友们大家好&#xff0c;最近想着怎么把网易云的歌保存到U盘&#xff0c;然后放到车上去听&#xff0c;然后辛辛苦苦搞了一宿&#xff0c;第二天拿到车上发现播放不了&#xff0c;根本就不认识 ncm 格式&#xff0c;我百度了一下&#xff0c;发现 ncm 是网易云的专用加密…

推荐一一款小众黑科技工具,低调使用建议收藏

wireshark是个啥就不多说了&#xff0c;非常流行的网络封包分析软件。 可以截取各种网络封包&#xff0c;显示网络封包的详细信息。 软件功能十分强大&#xff0c;操作也不复杂。 很多小友都在后台问能不能出一期完整的抓包分析贴&#xff0c;今天给你们安排上了哈。 01 W…

Kafka(二)【文件存储机制 生产者】

目录 一、Kafka 文件存储机制 二、Kafka 生产者 1、生产者消息发送流程 1.1、发送原理 2、异步发送 API 2.1、普通异步发送 案例演示 2.2、带回调函数的异步发送 2.3、同步发送 API 3、生产者分区 3.1、分区的好处 3.2、生产者发送消息的分区策略 &#xff08;1&am…

【Java数据结构 -- 队列:队列有关面试oj算法题】

队列、循环队列、用队列模拟栈、用栈模拟队列 1.队列1.1 什么是队列1.2 创建队列1.3 队列是否为空和获取队头元素 empty()peek()1.4 入队offer()1.5 出队&#xff08;头删&#xff09;poll() 2. 循环队列2.1 创建循环队列2.2 判断是否为空isEmpty()和满isFull()2.3 入队enQueue…