C++数据结构X篇_12_树的基本概念和存储

学习二叉树之前先学习树的概念。

1. 树的基本概念

之前所学均为线性表,线性表具有什么特点呢?
第一个节点只有一个后继结点,没有前驱,最后一个节点,只有一个前驱,没有后继,其他位置的节点都有前驱和后继。
线性表中每个节点之间都是一对一的关系,存储也就相对容易一些。

下图就是一个树的结构,每一个节点是一对多的关系,每一个节点都有一个双亲节点(父节点、爹妈节点),但是有多个后继,这就是树概念
在这里插入图片描述

1.1 树的定义

由一个或多个(n≥0)结点组成的有限集合 T,有且仅有一个结点称为根(root),当n>1时,其余的结点分为m(m≥0)个互不相交的有限集合 T1,T2,…,Tm。每个集合本身又是棵树,被称作这个根的子树。

1.2 树的特点

  • 非线性结构,有一个直接前驱,但可能有多个直接后继(1:n )
  • 树的定义具有递归性,树中还有树。
  • 树可以为空,即节点个数为 0。

1.3 若干术语

  • –>即根结点(没有前驱)
  • 叶子–>即终端结点(没有后继)
  • 森林–>指 m 棵不相交的树的集合(例如删除 A 后的子树个数)
  • 有序数–>结点各子树从左至右有序,不能互换(左为第一)
  • 无序树–>结点各子树可互换位置。
  • 双亲–>即上层的那个结点(直接前驱) parent
  • 孩子–>即下层结点的子树(直接后继) child
  • 兄弟–>同一双亲下的同层结点(孩子之间互称兄弟) sibling
  • 堂兄弟–>即双亲位于同一层的结点(但并非同一双亲)cousin
  • 祖先–>即从根到该结点所经分支的所有结点
  • 子孙–>即该结点下层子树中的任一结点
    在这里插入图片描述
  • 结点–>即树的数据元素
  • 结点的度–>结点挂接的子树数( 有几个直接后继就是几度 )
  • 结点的层次–>从根到该结点的层数( 根结点算第一层 )
  • 终端结点–>即度为 0 的结点,即叶子
  • 分支结点–>除树根以外的结点( 也称为内部结点)
  • 树的度–>所有结点度中的最大值( Max(各结点的度》)
  • 树的深度(或高度)–>所有结点中最大的层数( Max(各结点的层次))
    上图中的结点数= 13,树的度= 3,树的深度= 4

2. 树的表示法

2.1 图形表示法

事物之间的逻辑关系可以通过数的形式很直观的表示出来,如下图:
在这里插入图片描述

2.2 广义表表示法

在这里插入图片描述
用广义表表示法表示上图:
中国( 河北( 保定,石家庄 ),广东( 广州,东莞 ),山东( 青岛,济南 ) )
根作为由子树森林组成的表的名字写在表的左边。

3. 树的存储

前面我们学习了顺序存储链式存储,以下面的树为例,如何进行存储呢?也可以使用顺序存储:A,B,C,D...M,放到数组中,但是你不知道谁是谁儿子和爹吗?
在这里插入图片描述

3.1 双亲表示法:保存父节点关系

了解即可

struct Node {
   int data; //节点值
   int parent; //父节点下标
}

为了找到某一个节点的孩子是谁,需要进行遍历查看父节点信息。
上面的方法是站在爹妈的角度去看的

3.2 孩子表示法

了解即可
以孩子成长的角度来看就是孩子表示法。以链式去保存,需要保存孩子的信息,但是孩子数量不固定,对于ChildNode*就不知道定义几个指针了,定义的多了就会产生多余指针。

可以通过链表将孩子节点进行保存,这样就可以通过链表将节点进行保存

struct ChildNode {
   int data; //节点值
   ChildNode*
}
childNode D;
D.ChildNode*;

3.3 左孩子右兄弟表示法

不管是什么样的树,都可以转换为二叉树

第一个节点A,左孩子为B,右兄弟没有,B的左孩子E,右兄弟C,E的左孩子K,右兄弟F,K的左孩子没有,右兄弟L,到F,左孩子和右兄弟没有,到C,左孩子G,右兄弟D,D的左孩子H,右兄弟没有,到H,左孩子M,右兄弟I,I的左孩子没有,右兄弟J。

在这里插入图片描述
通过左孩子右兄弟之后就转换为这种树,每个节点有0到2个孩子,这种树称为二叉树。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
引用\[1\]:在C++中,数据结构是指一种组织和存储数据的方式。在这段代码中,使用了结构体来定义一个节点(node),节点包含了两个整数变量x和y。结构体中还重载了小于号运算符,用于比较节点的大小。主函数中使用了优先队列(priority_queue)来存储节点,并按照节点的x值从大到小进行排序。然后通过遍历优先队列,输出节点的x和y值。\[1\] 引用\[2\]:在C++中,vector是一种动态数组容器。它可以根据需要自动调整大小,并且支持多种构造函数。例如,可以使用默认构造函数创建一个空的vector,也可以使用拷贝构造函数将一个vector的元素拷贝给另一个vector。另外,还可以使用带有两个迭代器参数的构造函数,将一个区间内的元素拷贝给vector,或者使用带有一个整数参数和一个元素参数的构造函数,将指定数量的相同元素拷贝给vector。\[2\] 引用\[3\]:这段代码是一个关于图的遍历的例题。首先,根据输入的节点数量n和边的数量m,使用并查集来判断图是否联通。然后,统计图中奇点的数量,如果奇点的数量为0或者2,则存在欧拉回路。最后,根据判断结果输出相应的结果。\[3\] 综上所述,C++中的数据结构基本概念包括使用结构体来定义节点,使用优先队列来排序节点,使用vector来存储动态数组,以及使用并查集来判断图的连通性和欧拉回路的存在性。 #### 引用[.reference_title] - *1* *2* [C++之STL基础概念、容器、数据结构](https://blog.csdn.net/Pxx520Tangtian/article/details/126764518)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control,239^v4^insert_chatgpt"}} ] [.reference_item] - *3* [c++数据结构-图(详解附算法代码,一看就懂)](https://blog.csdn.net/m0_64036070/article/details/128737229)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control,239^v4^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

十月旧城

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值