数据结构--树

一、树的概念

树是由n(n≥0)个节点组成的有限集合,它满足以下条件:
 
1. 当n=0时,称为空树
2. 当n>0时,有且仅有一个特定的节点称为根节点(root)
3. 其余节点可分为m(m≥0)个互不相交的有限集合,每个集合本身又是一棵树,称为根的子树(subtree)
 
二、基本术语
 
- 节点(Node): 树的基本单位,包含数据项和指向其他节点的指针
- 边(Edge): 连接两个节点的线
- 根节点(Root): 树的最顶层节点,没有父节点
- 父节点(Parent): 一个节点的直接上层节点
- 子节点(Child): 一个节点的直接下层节点
- 叶子节点(Leaf): 没有子节点的节点
- 内部节点(Internal Node): 至少有一个子节点的节点
- 度(Degree): 一个节点拥有的子节点数目
- 树的度: 树中所有节点度的最大值
- 层次(Level): 根节点为第1层,其子节点为第2层,以此类推
- 高度/深度(Height/Depth): 树中节点的最大层次数
- 兄弟节点(Sibling): 具有相同父节点的节点
- 祖先节点(Ancestor): 从根到该节点路径上的所有节点
- 后代节点(Descendant): 某节点子树中的所有节点
 
三、树的分类
 
1. 二叉树(Binary Tree): 每个节点最多有两个子节点
- 满二叉树
- 完全二叉树
- 二叉搜索树(BST)
- 平衡二叉树(AVL树)
- 红黑树
2. 多叉树: 每个节点可以有多个子节点
- B树
- B+树
- Trie树(字典树)
3. 其他特殊树结构
- 堆(Heap)
- 哈夫曼树
- 线段树
- 树状数组
 
四、树的存储表示
 
1. 链式存储
- 每个节点包含数据域和指针域
- 指针指向子节点
2. 顺序存储
- 使用数组存储
- 对于完全二叉树特别有效
 3.树的简单实现

#include <iostream>
#include <queue>
#include <algorithm>
using namespace std;// 定义树节点类模板
template <typename T>
class TreeNode
{
public:T data;  // 节点存储的数据TreeNode* firstChild;  // 指向该节点的第一个孩子节点TreeNode* nextSibling; // 指向该节点的下一个兄弟节点// 构造函数,初始化节点数据,将孩子和兄弟指针置为空TreeNode(T val) : data(val), firstChild(nullptr), nextSibling(nullptr) {}
};// 定义树类模板
template <typename T>
class Tree
{
private:TreeNode<T>* root;  // 树的根节点// 递归清空树的辅助函数void clear(TreeNode<T>* node){if (node == nullptr) return;  // 如果节点为空,直接返回clear(node->firstChild);  // 递归清空当前节点的第一个孩子节点及其子树clear(node->nextSibling); // 递归清空当前节点的下一个兄弟节点及其子树delete node;  // 释放当前节点的内存}public:// 构造函数,初始化根节点为空Tree() : root(nullptr) {}// 析构函数,调用 clear 函数清空整棵树~Tree() { clear(root); }// 创建一个新的树节点,返回新节点的指针TreeNode<T>* createNode(T data){return new TreeNode<T>(data);}// 设置树的根节点void setRoot(TreeNode<T>* node){root = node;}// 获取树的根节点TreeNode<T>* getRoot() const{return root;}// 向指定父节点插入一个孩子节点void insertChild(TreeNode<T>* parent, TreeNode<T>* child){if (parent == nullptr) return;  // 如果父节点为空,直接返回if (parent->firstChild == nullptr){// 如果父节点没有第一个孩子节点,将该孩子节点设为第一个孩子parent->firstChild = child;}else{// 否则,找到父节点孩子链表的末尾,将该孩子节点插入到末尾TreeNode<T>* sibling = parent->firstChild;while (sibling->nextSibling != nullptr){sibling = sibling->nextSibling;}sibling->nextSibling = child;}}// 前序遍历树void preOrderTraversal(TreeNode<T>* node){if (node == nullptr) return;  // 如果节点为空,直接返回cout << node->data << " ";  // 访问当前节点的数据preOrderTraversal(node->firstChild);  // 递归前序遍历当前节点的第一个孩子节点及其子树preOrderTraversal(node->nextSibling); // 递归前序遍历当前节点的下一个兄弟节点及其子树}// 后序遍历树void postOrderTraversal(TreeNode<T>* node){if (node == nullptr) return;  // 如果节点为空,直接返回postOrderTraversal(node->firstChild);  // 递归后序遍历当前节点的第一个孩子节点及其子树cout << node->data << " ";  // 访问当前节点的数据postOrderTraversal(node->nextSibling); // 递归后序遍历当前节点的下一个兄弟节点及其子树}// 层序遍历树void levelOrderTraversal(){if (root == nullptr) return;  // 如果根节点为空,直接返回queue<TreeNode<T>*> q;  // 定义一个队列用于层序遍历q.push(root);  // 将根节点入队while (!q.empty()){TreeNode<T>* current = q.front();  // 获取队首节点q.pop();  // 队首节点出队cout << current->data << " ";  // 访问当前节点的数据// 将当前节点的所有孩子节点依次入队TreeNode<T>* child = current->firstChild;while (child != nullptr){q.push(child);child = child->nextSibling;}}}// 获取树的高度int getHeight(TreeNode<T>* node){if (node == nullptr) return 0;  // 如果节点为空,高度为 0int maxHeight = 0;  // 初始化最大子树高度为 0TreeNode<T>* child = node->firstChild;while (child != nullptr){// 递归计算每个子树的高度,并更新最大子树高度maxHeight = max(maxHeight, getHeight(child));child = child->nextSibling;}return maxHeight + 1;  // 当前节点的高度为最大子树高度加 1}// 计算树中节点的总数int countNodes(TreeNode<T>* node){if (node == nullptr) return 0;  // 如果节点为空,节点数为 0// 当前节点及其子树的节点总数为当前节点加上其第一个孩子节点及其子树的节点数,再加上其兄弟节点及其子树的节点数return 1 + countNodes(node->firstChild) + countNodes(node->nextSibling);}// 在树中查找值为 value 的节点TreeNode<T>* findNode(TreeNode<T>* node, T value){if (node == nullptr) return nullptr;  // 如果节点为空,返回空指针if (node->data == value) return node;  // 如果当前节点的值等于要查找的值,返回当前节点的指针// 先在当前节点的第一个孩子节点及其子树中查找TreeNode<T>* found = findNode(node->firstChild, value);if (found != nullptr) return found;// 若未找到,再在当前节点的下一个兄弟节点及其子树中查找return findNode(node->nextSibling, value);}// 打印树的结构,level 表示当前节点的层级void printTree(TreeNode<T>* node, int level = 0){if (node == nullptr) return;  // 如果节点为空,直接返回// 根据当前节点的层级打印相应数量的空格for (int i = 0; i < level; ++i){cout << "  ";}cout << node->data << endl;  // 打印当前节点的数据// 递归打印当前节点的第一个孩子节点及其子树,层级加 1printTree(node->firstChild, level + 1);// 递归打印当前节点的下一个兄弟节点及其子树,层级不变printTree(node->nextSibling, level);}
};int main()
{Tree<char> tree;  // 创建一个存储字符类型数据的树对象// 创建树的各个节点TreeNode<char>* A = tree.createNode('A');TreeNode<char>* B = tree.createNode('B');TreeNode<char>* C = tree.createNode('C');TreeNode<char>* D = tree.createNode('D');TreeNode<char>* E = tree.createNode('E');TreeNode<char>* F = tree.createNode('F');TreeNode<char>* G = tree.createNode('G');// 构建树的结构tree.setRoot(A);  // 设置 A 为根节点tree.insertChild(A, B);  // 将 B 插入为 A 的孩子节点tree.insertChild(A, C);  // 将 C 插入为 A 的孩子节点tree.insertChild(A, D);  // 将 D 插入为 A 的孩子节点tree.insertChild(B, E);  // 将 E 插入为 B 的孩子节点tree.insertChild(B, F);  // 将 F 插入为 B 的孩子节点tree.insertChild(D, G);  // 将 G 插入为 D 的孩子节点// 打印树的结构cout << "Tree structure:" << endl;tree.printTree(tree.getRoot());cout << endl;// 进行各种遍历测试cout << "Pre-order traversal: ";tree.preOrderTraversal(tree.getRoot());cout << endl;cout << "Post-order traversal: ";tree.postOrderTraversal(tree.getRoot());cout << endl;cout << "Level-order traversal: ";tree.levelOrderTraversal();cout << endl;// 测试其他功能cout << "Tree height: " << tree.getHeight(tree.getRoot()) << endl;cout << "Total nodes: " << tree.countNodes(tree.getRoot()) << endl;// 测试查找节点功能char searchValue = 'F';TreeNode<char>* found = tree.findNode(tree.getRoot(), searchValue);if (found){cout << "Found node: " << found->data << endl;}else{cout << "Node " << searchValue << " not found" << endl;}return 0;
}


五、树的应用
 
- 文件系统目录结构
- 数据库索引
- 网络路由算法
- 决策树(机器学习)
- XML/HTML文档对象模型(DOM)
- 组织结构图
- 游戏中的场景图
 
树结构因其高效的查找、插入和删除操作(O(log n)时间复杂度)而被广泛应用于各种算法和系统中。

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

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

相关文章

Linux `ifconfig` 指令深度解析与替代方案指南

Linux `ifconfig` 指令深度解析与替代方案指南 一、核心功能与现状1. 基础作用2. 版本适配二、基础语法与常用操作1. 标准语法2. 常用操作速查显示所有接口信息启用/禁用接口配置IPv4地址修改MAC地址(临时)三、高级配置技巧1. 虚拟接口创建2. MTU调整3. 多播配置4. ARP控制四…

什么是分布式光伏系统?屋顶分布式光伏如何并网?

政策窗口倒计时&#xff01;分布式光伏如何破局而立&#xff1f; 2025年&#xff0c;中国分布式光伏行业迎来关键转折&#xff1a; ▸ "430"落幕——抢装潮收官&#xff0c;但考验才刚开始&#xff1b; ▸ "531"生死线——新增项目全面市场化交易启动&…

Cluster Interconnect in Oracle RAC

Cluster Interconnect in Oracle RAC (文档 ID 787420.1)​编辑转到底部 In this Document Purpose Scope Details Physical Layout of the Private Interconnect Why Do We Need a Private Interconnect ? Interconnect Failure Interconnect High Availability Private Inte…

.Net HttpClient 使用准则

HttpClient 使用准则 System.Net.Http.HttpClient 类用于发送 HTTP 请求以及从 URI 所标识的资源接收 HTTP 响应。 HttpClient 实例是应用于该实例执行的所有请求的设置集合&#xff0c;每个实例使用自身的连接池&#xff0c;该池将其请求与其他请求隔离开来。 从 .NET Core …

【PostgreSQL】数据库主从库备份与高可用部署

文章目录 一、架构设计原理二、部署清单示例2.1 StatefulSet配置片段2.2 Service配置三、配置详解3.1 主节点postgresql.conf3.2 从节点配置四、初始化流程4.1 创建复制用户4.2 配置pg_hba.conf五、故障转移示例5.1 自动切换脚本5.2 手动提升从节点六、监控与维护6.1 关键监控指…

JavaScript 数组去重:11 种方法对比与实战指南

文章目录 前言一、使用 Set 数据结构二、使用 filter indexOf三、使用 reduce 累加器四、双重 for 循环五、利用对象属性唯一性六、先排序后去重七、使用 Map 数据结构八、使用 includes 方法九、优化处理 NaN 的 filter 方法十、利用 findIndex十一.利用Set和展开运算符处理多…

ai agent(智能体)开发 python3基础14:在python 中 总能看到方法里面套方法,那什么时候用这种方式合适呢?

让人头疼的方法嵌套还是要去了解的 在 Python 中&#xff0c;方法内部嵌套方法&#xff08;即在类的方法中定义另一个函数&#xff09;是一种常见的代码组织技巧&#xff0c;它可以在特定场景下带来以下好处&#xff1a; 1. 代码复用与逻辑封装 如果某个方法内部有重复的逻辑…

Yocto项目实战经验总结:从入门到高级的全面概览

本文面向开发者和实际项目经验者&#xff0c;分享经过大量实战积累的 Yocto 项目工程经验和基础技巧。本文简明但精彩&#xff0c;应用和观察相结合&#xff0c;充分适合做为全面进阶 Yocto 项目开发的实用指南。 一、入门理解&#xff1a;Yocto 是什么&#xff1f;规划如何开始…

添加物体.

在cesium中我们可以添加物体进入地图.我们以广州塔为例 //生成广州塔的位置var position2 Cesium.Cartesian3.fromDegrees(113.3191,23.109,100)viewer.camera.setView({//指定相机位置destination: position2, 运行后如图 我们使用cesium官网提供的代码为广州塔在地图上标点…

正则表达式非捕获分组?:

一个使用 Java 正则表达式的具体例子&#xff0c;展示了 (ab) 和 (?:ab) 的不同&#xff1a; 示例 1&#xff1a;使用 (ab)&#xff08;捕获分组&#xff09; import java.util.regex.*; public class RegexExample { public static void main(String[] args) { …

ragflow报错:KeyError: ‘\n “序号“‘

环境&#xff1a; ragflowv 0.17.2 问题描述&#xff1a; ragflow报错&#xff1a;KeyError: ‘\n “序号”’ **1. 推荐表&#xff08;输出json格式&#xff09;** [{"},{},{"},{} ]raceback (most recent call last): May 08 20:06:09 VM-0-2-ubuntu ragflow-s…

Spring Boot-8启动涉及的监听器(扩展点)

从出现时间上看&#xff1a; org.springframework.context.ApplicationListener&#xff0c;Spring 1.0开始出现 org.springframework.context.ApplicationContextInitializer&#xff0c;Spring 3.1开始出现 org.springframework.boot.SpringApplicationRunListener&#x…

如何启动vue项目及vue语法组件化不同标签应对的作用说明

如何启动vue项目及vue语法组件化不同标签应对的作用说明 提示&#xff1a;帮帮志会陆续更新非常多的IT技术知识&#xff0c;希望分享的内容对您有用。本章分享的是node.js和vue的使用。前后每一小节的内容是存在的有&#xff1a;学习and理解的关联性。【帮帮志系列文章】&…

思考:(linux) tmux 超级终端快速入门的宏观思维

tmux 工具集合 GitHub - rothgar/awesome-tmux: A list of awesome resources for tmux 要点&#xff1a; 习惯性思维的变换与宿主机之间的双向复制、粘贴手动备份全部窗口&#xff0c;以及还原自定义窗格提示信息TPM 插件的安装思想别名 在有些场景里&#xff0c;可能无法…

Python实例题:Python协程详解公开课

目录 Python实例题 题目 课程目标 课程内容规划 1. 课程开场&#xff08;5 分钟&#xff09; 2. 基础概念讲解&#xff08;15 分钟&#xff09; 并发与并行&#xff1a; 线程与进程&#xff1a; 3. Python 协程的实现方式&#xff08;20 分钟&#xff09; 生成器实现…

AI时代的数据可视化:未来已来

你有没有想过&#xff0c;数据可视化在未来会变成什么样&#xff1f;随着人工智能&#xff08;AI&#xff09;的飞速发展&#xff0c;数据可视化已经不再是简单的图表和图形&#xff0c;而是一个充满无限可能的智能领域。AI时代的可视化不仅能自动解读数据&#xff0c;还能预测…

强化学习PPO算法学习记录

1. 四个模型&#xff1a; Policy Model&#xff1a;我们想要训练的目标语言模型。我们一般用SFT阶段产出的SFT模型来对它做初始化。Reference Model&#xff1a;一般也用SFT阶段得到的SFT模型做初始化&#xff0c;在训练过程中&#xff0c;它的参数是冻结的。Ref模型的主要作用…

边缘计算从专家到小白

“云-边-端”架构 “云” &#xff1a;传统云计算的中心节点&#xff0c;是边缘计算的管控端。汇集所有边缘的感知数据、业务数据以及互联网数据&#xff0c;完成对行业以及跨行业的态势感知和分析。 “边” &#xff1a;云计算的边缘侧&#xff0c;分为基础设施边缘和设备边缘…

Windows:Powershell的使用

文章目录 零、格式化输出命令1、Format-List&#xff08;别名&#xff1a;fl&#xff09; 一、服务管理SC命令二、软件管理命令三、权限管理命令1、Get-Acl2、Set-Acl 总结 零、格式化输出命令 1、Format-List&#xff08;别名&#xff1a;fl&#xff09; 可通过管道符传递对象…

实现在h5中添加日历提醒:safari唤起系统日历,其它浏览器跳转google日历

需求&#xff1a;点击按钮后&#xff0c;将设定的一些信息插入到系统日历的日程安排中。 调研过程 先google了一段时间&#xff0c;了解该需求大概的实现方式。可以创建日历文件&#xff0c;在点击的时候下载该日历文件&#xff0c;看起来还比较复杂&#xff0c;并且由于不具…