数据结构详细笔记——二叉树

文章目录

  • 二叉树的定义和基本术语
  • 特殊的二叉树
    • 满二叉树
    • 完全二叉树
    • 二叉排序树
    • 平衡二叉树
  • 二叉树的常考性质
  • 完全二叉树的常考性质
  • 二叉树的存储结构
    • 顺序存储
    • 链式存储
  • 二叉树的先中后序遍历
    • 先序遍历(空间复杂度:O(h))
    • 中序遍历
    • 后序遍历
    • 应用
  • 二叉树的层序遍历
  • 由遍历序列构造二叉树
  • 线索二叉树
    • 线索二叉树的存储结构
    • 二叉树的线索化
    • 二叉树的线索化


二叉树的定义和基本术语

二叉树的基本概念

二叉树是n(n>=0)个结点的有限集合
①或者为空的二叉树,即n=0
②或者由一个根结点和两个互不相交的被称为根的左子树和右子树组成,左子树和右子树又分别是一颗二叉树

特点:
1、每一个结点至多只有两棵子树
2、左右子树不能颠倒(二叉树是有序树)

注意区别:度为2的有序树与二叉树的区别
度为2的树:肯定是非空树,所有结点的度<=2,至少有一个结点的度为2
二叉树:可以为空树,所有的结点只要<=2就可

特殊的二叉树

满二叉树

定义:一颗高度为n,且有 2ⁿ-1 个结点的二叉树
在这里插入图片描述

特点:
1、只有最后一层有叶子结点
2、不存在度为1的结点(只存在度为0或者2的结点)
3、按层序从1开始编号,结点 i 的左孩子为 2i,右孩子为 2i+1,结点 i 的父节点为 ⌊ i/2 ⌋;

完全二叉树

定义:当且仅当其每个结点都与高度为 h 的满二叉树中的编号为 1~n 的结点一一对应时,称为完全二叉树
在这里插入图片描述
特点:
1、只有最后两层有叶子结点
2、最多只有一个度为1的结点
3、按层序从1开始编号,结点 i 的左孩子为 2i,右孩子为 2i+1,结点 i 的父节点为 ⌊ i/2 ⌋;(如果有的话)
4、i <= ⌊ n/2 ⌋ 为分支结点,i > ⌊ n/2 ⌋ 为叶子结点

如果某结点只有一个孩子,那么这个孩子一定是左孩子

二叉排序树

定义:一颗二叉树或者空二叉树具有如下性质的称为二叉排序树
1、左子树上所有结点的关键字均小于根结点的关键字
2、右子树上所有结点的关键字均大于根结点的关键字
3、左子树和右子树又各是一棵二叉排序树

二叉排序树可用于元素的排序和搜索

平衡二叉树

定义:树上任一结点左子树右子树深度之差不超过1

平衡二叉树能有更高的搜索效率

二叉树的常考性质

考点一:设非空二叉树中度为0、1、2的结点个数分别为n0、n1、n2,则 n0 = n2 + 1; ( 叶子结点比二分支结点多一个 )

假设树中结点总数为n,则
① n = n0 + n1 +n2
② n = n1 + 2*n2 + 1 (树的结点数 = 总度数 + 1)
②-①得 n0 = n2 + 1

考点二:二叉树第 i 层至多有 2ⁱ⁻¹ 个结点(i>=1)
源于: m 叉树第 i 层至多有 mⁱ⁻¹ 个结点(i>=1)

考点三:高度为 n 的二叉树至多有 2ⁿ - 1 个结点(满二叉树)
源于: 高度为 n 的 m 叉树至多有 mⁿ - 1/m-1 个结点

完全二叉树的常考性质

在这里插入图片描述

常考考点2:对于完全二叉树,可以由结点数 n 推出度为0、1、2的结点个数为n0、n1、n2
完全二叉树最多只有一个度为1的结点,所以
n1 = 0或1
n0 = n2 + 1 ----> n0 + n2 = 2*n2 + 1 一定是奇数

若完全二叉树有 2k 个(偶数)个结点,则 n = n1 + n2 + n3,由上面得 n0 + n2 一定是奇数,所以只有 n1 = 1 时, n 才会是偶数
所以 n1 = 1,n0 = k,n2 = k-1

若完全二叉树有 2k-1 个(奇数)个结点,则 n = n1 + n2 + n3,由上面得 n0 + n2 一定是奇数,所以只有 n1 = 0 时, n 才会是奇数
所以 n1 = 0,n0 = k,n2 = k-1

二叉树的存储结构

顺序存储

定义一个长度为 MaxSize 的数组t,按照从上至下,从左至右的顺序依次存储完全二叉树中的各个结点

#define MaxSize 100
struct TreeNode{ElemType value;   // 结点中的数据元素bool isEmpty;     // 结点是否为空
}// 初始化时所有结点标记为空
for(int i = 0; i < MaxSize; i++){t[i].isEmpty = true;
}TreeNode t[MaxSize]

几个重要的常考的基本操作
i 的左孩子 ———— 2i
i 的右孩子 ———— 2i+1
i 的父节点 ———— ⌊ i/2 ⌋
i 所在的层次 ———— 在这里插入图片描述

若完全二叉树中共有n个结点,则
判断 i 是否有左孩子? ———— 2i <= n
判断 i 是否有右孩子? ———— 2i+1 <= n
判断 i 是否是叶子/分支结点? ———— i > ⌊ n/2 ⌋

注意:如果不是完全二叉树,依然按照层序将各个结点顺序存储,那么无法从结点编号反映出以上这些逻辑关系,所以一定要把二叉树的结点编号与完全二叉树对应起来,但是会导致出现很多空值的存储单元

最坏情况:高度为 n 且只有 n 个结点的单支树(所有结点只有右孩子),也至少需要 2ⁿ - 1 个存储单元

结论:二叉树的顺序存储结构,只适合存储完全二叉树

链式存储

typedef struct BiTNode{ElemType data;struct BiTNode *lchild,*rchild;  // 左右孩子指针
}BiTNode,*BiTree;

假设有n个结点,则有2n个指针域,除了根结点,每一个结点头上都会连接一个指针域,那么就有n-1个有值的指针域,得出就会有n+1个空指针域
n个结点的二叉链表共有n+1个空链域

根据实际需求决定要不要加父结点指针(三叉链表——方便找父结点)

typedef struct BiTNode{ElemType data;struct BiTNode *lchild,*rchild;  // 左右孩子指针struct BiTNode *parent;          // 父结点指针
}BiTNode,*BiTree;

二叉树的先中后序遍历

  1. 先序遍历:根左右
  2. 中序遍历:左根右
  3. 后序遍历:左右根

其中这三种遍历方式与前中后缀表达式的关系
先序遍历——>前缀表达式
中序遍历——>中缀表达式(需要加界限符)
后序遍历——>后缀表达式

先序遍历(空间复杂度:O(h))

void PreOrder(BiTree T){if(T!=NULL){visit(T);     //访问根结点PreOrder(T->lchild);   // 递归遍历左子树PreOrder(T->rchild);   // 递归遍历右子树}
}

中序遍历

void PreOrder(BiTree T){if(T!=NULL){PreOrder(T->lchild);   // 递归遍历左子树visit(T);     //访问根结点PreOrder(T->rchild);   // 递归遍历右子树}
}

后序遍历

void PreOrder(BiTree T){if(T!=NULL){PreOrder(T->lchild);   // 递归遍历左子树PreOrder(T->rchild);   // 递归遍历右子树visit(T);     //访问根结点}
}

应用

求树的深度

int treeDepth(BiTree T){if(T == NULL)return 0;else{int l = treeDepth(T->lchild);int r = treeDepth(T->rchild);// 树的深度=Max(左子树深度,右子树深度) + 1return l>r ? l+1 : r+1;}
}

二叉树的层序遍历

算法思想:
①初始化一个辅助队列
②根结点入队
③若队列非空,则队头结点出队,访问该结点,并将其左、右孩子插入队尾(如果有的话)
④重复③直至队列为空

// 二叉树的结点(链式存储)
typedef struct BiTNode{char data;struct BiTNode *lchild,*rchild;
}BiTNode,*BiTree;// 链式队列结点
typedef struct LinkNode{BiTNode *data;  // 存指针,而不是结点struct LinkNode *next;
}LinkNode;typedef struct{LinkNode *front,*rear;  // 队头队尾
}LinkQueue;// 层序遍历
void LevelOrder(BiTree T){LinkQueue(Q);InitQueue(Q);  // 初始化辅助队列BiTree p;EnQueue(Q,T);  // 将根结点入队while(!IsEmpty(Q)){DeQueue(Q,p);  // 队头结点出队visit(p);       // 访问出队结点if(p->lchild!=NULL)EnQueue(Q,(p->lchild);  // 左孩子入队if(p->rchild!=NULL)EnQueue(Q,(p->rchild);  // 右孩子入队}
}

由遍历序列构造二叉树

若只给出一颗二叉树的前中后序遍历序列中的一种,不能唯一确定一棵二叉树
能构造二叉树的遍历序列组合
1、前序+中序
2、后序+中序
3、层序+中序

线索二叉树

问题1:如何找到指定结点在中序遍历序列中的前驱?
问题2:如何找到结点的中序后继?
找前驱和后继很不方便,遍历操作必须从根开始

还记得之前学到的,n个结点的二叉树,有n+1个空链域,可以用来记录前驱、后继的信息

线索二叉树的存储结构

typedef struct ThreadNode{ElemType data;struct ThreadNode *lchild,*rchild;int ltag,rtag;    // 左右线索标志
}ThreadNode,*ThreadTree;
// tag=0 表示指针指向孩子
// tag=1 表示指针是线索

在这里插入图片描述

在这里插入图片描述

二叉树的线索化

中序线索化

// 线索二叉树结点
typedef struct ThreadNode{ElemType data;struct ThreadNode *lchild,*rchild;int ltag,rtag;    // 左右线索标志
}ThreadNode,*ThreadTree;void visit(ThreadNode *q){if(q->lchild==NULL){ //左子树为空,建立前驱线索q->lchild=pre;q->ltag=1;}if(pre!=NULL&&pre->rchild==NULL){ //前驱结点的右孩子为空,建立前驱结点的后驱线索pre->rchild=q;pre->rtag=1;}pre=q;
}//中序遍历二叉树,一边遍历一边线索化
void InThread(ThreadTree T){if(T!=NULL){InThread(T->lchild);  // 中序遍历左子树visit(T);             // 访问根结点InThread(T->rchild);  // 中序遍历右子树}
}// 全局变量 pre,指向当前访问结点的前驱
ThreadNode *pre=NULL;// 中序线索化二叉树T
void CreateInThread(ThreadTree T){pre=NULL;   // pre初始化为NULLif(T!=NULL){InThread(T)  //中序线索化二叉树if(pre->rchild==NULL)pre->rtag=1;  // 处理遍历的最后一个点}
}

先序线索化(与中序遍历有一点不一样:为了防止重复指向,需要判断lchild不是前驱线索)

// 线索二叉树结点
typedef struct ThreadNode{ElemType data;struct ThreadNode *lchild,*rchild;int ltag,rtag;    // 左右线索标志
}ThreadNode,*ThreadTree;void visit(ThreadNode *q){if(q->lchild==NULL){ //左子树为空,建立前驱线索q->lchild=pre;q->ltag=1;}if(pre!=NULL&&pre->rchild==NULL){ //前驱结点的右孩子为空,建立前驱结点的后驱线索pre->rchild=q;pre->rtag=1;}pre=q;
}//先序遍历二叉树,一边遍历一边线索化
void PreThread(ThreadTree T){if(T!=NULL){visit(T);             // 访问根结点if(T->ltag==0){         // 与中序遍历不一样的地方,为了防止重复指向,需要判断lchild不是前驱线索PreThread(T->lchild);  // 与中序遍历不一样的地方,为了防止重复指向,需要判断lchild不是前驱线索PreThread(T->rchild);}
}// 全局变量 pre,指向当前访问结点的前驱
ThreadNode *pre=NULL;// 先序线索化二叉树T
void CreateInThread(ThreadTree T){pre=NULL;   // pre初始化为NULLif(T!=NULL){InThread(T)  //先序线索化二叉树if(pre->rchild==NULL)pre->rtag=1;  // 处理遍历的最后一个点}
}

后序线索化(和中序差不多)

// 线索二叉树结点
typedef struct ThreadNode{ElemType data;struct ThreadNode *lchild,*rchild;int ltag,rtag;    // 左右线索标志
}ThreadNode,*ThreadTree;void visit(ThreadNode *q){if(q->lchild==NULL){ //左子树为空,建立前驱线索q->lchild=pre;q->ltag=1;}if(pre!=NULL&&pre->rchild==NULL){ //前驱结点的右孩子为空,建立前驱结点的后驱线索pre->rchild=q;pre->rtag=1;}pre=q;
}//后序遍历二叉树,一边遍历一边线索化
void InThread(ThreadTree T){if(T!=NULL){InThread(T->lchild);  // 后序遍历左子树InThread(T->rchild);  // 后序遍历右子树visit(T);             // 访问根结点}
}// 全局变量 pre,指向当前访问结点的前驱
ThreadNode *pre=NULL;// 后序线索化二叉树T
void CreateInThread(ThreadTree T){pre=NULL;   // pre初始化为NULLif(T!=NULL){InThread(T)  //后序线索化二叉树if(pre->rchild==NULL)pre->rtag=1;  // 处理遍历的最后一个点}
}

二叉树的线索化

中序线索二叉树中找到指定结点*p的中序后继next
① 若 p->rtag=1, 则 next = p->rchild
② 若 p->rtag=0, 则 next = p的右子树中最左下结点

// 找到以p为根的子树中,第一个被中序遍历的结点
ThreadNode *Firstnode(ThreadNode *p){// 循环找到最左下结点(不一定是叶结点)while(p->ltag==0) p=p->lchild;return p;
}// 在中序线索二叉树中找到结点p的后继结点
ThreadNode *Nextnode(ThreadNode *p){// 右子树下最左的结点if(p->rtag==0) return Firstnode(p->rchild);esle return p->rchild;   // rtag ==1 直接返回后继线索
}// 对中序线索二叉树进行中序遍历(利用线索实现的非递归算法)空间复杂O(1)
void Inorder(TreadNode *T){for(ThreadNode *p=Firstnode(T);p!=NULL;p=Nextnode(p)){visit(p);}
}

中序线索二叉树中找到指定结点*p的中序前驱pre
① 若 p->ltag=1, 则 pre= p->rchild
② 若 p->rlag=0, 则 pre= p的左子树中最右下结点

// 找到以p为根的子树中,最后一个被中序遍历的结点
ThreadNode *Lastnode(ThreadNode *p){// 循环找到最左下结点(不一定是叶结点)while(p->rtag==0) p=p->rchild;return p;
}// 在中序线索二叉树中找到结点p的前驱结点
ThreadNode *Prenode(ThreadNode *p){// 左子树中最右下结点if(p->ltag==0) return Lastnode(p->lchild);esle return p->lchild;   // ltag ==1 直接返回前驱线索
}// 对中序线索二叉树进行逆向中序遍历
void RevInorder(TreadNode *T){for(ThreadNode *p=Lastnode(T);p!=NULL;p=Prenode(p)){visit(p);}
}

在先序线索二叉树中找到指定结点*p的先序后继next
① 若 p->rtag=1, 则 next = p->rchild
② 若 p->rtag=0,
1、若p有左孩子,则先序后继为左孩子
2、若p没有左孩子,则先序后继为右孩子

在先序线索二叉树中找到指定结点*p的先序前驱pre
① 若 p->ltag=1, 则 next = p->lchild
② 若 p->ltag=0, 需要从头开始遍历

在后序线索二叉树中找到指定结点*p的后序前驱pre
① 若 p->ltag=1, 则 pre= p->lchild
② 若 p->ltag=0,
1、若p有右孩子,则后序前驱为右孩子
2、若p没有右孩子,则后序前驱为左孩子

在后序线索二叉树中找到指定结点*p的后序后继next
① 若 p->rtag=1, 则 next = p->rchild
② 若 p->rtag=0, 需要从头开始遍历

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

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

相关文章

有奖快来抱走全新HUAWEI WATCH GT4

亲爱的openGauss用户&#xff0c; 为了给您提供更好的社区体验&#xff0c;现诚邀您参与openGauss社区满意度问卷调研。您的每一个宝贵建议都是我们进步的方向。 手机扫描二维码即可填写问卷&#xff0c;请根据您真实的体验情况填写问卷&#xff0c;问卷反馈越真实有效越有机…

vue3写nav滚动事件中悬停在顶部

1. 防抖类Animate, 使用requestAnimationFrame代替setTimeout 也可以使用节流函数, lodash有现成的防抖和节流方法 _.debounce防抖 _.throttle节流 export default class Animate {constructor() {this.timer null;}start (fn) > {if (!fn) {throw new Error(需要执行…

【面试】什么是霍桑效应,在项目中具体要怎样使用霍桑效应呢?

其实霍桑效应&#xff08;Hawthorne Effect&#xff09;是一种很经典的组织行为学现象&#xff0c;它阐述了被观察者会因知晓自己被观察而改变自己的行为。在项目管理中&#xff0c;我们可以通过霍桑效应来激励团队提高工作效率。具体运用可以从以下几个方面&#xff1a; 告知…

亲测解决Pytorch TypeError: object of type ‘numpy.int64‘ has no len()

这个问题是小虎在初始化自适应平均池化的时候遇到的&#xff0c;解决方法是限制初始化时池化大小的类型。 问题原文 Exception has occurred: TypeError object of type numpy.int64 has no len()File "D:\Complier\LEF\lib\model\segmentation\heads\modules\fgModules…

使用 TensorFlow SSD 网络进行对象检测

使用 TensorFlow SSD 网络进行对象检测 目录 描述这个示例是如何工作的&#xff1f; 处理输入图准备数据sampleUffSSD 插件验证输出TensorRT API 层和操作 先决条件运行示例 示例 --help 选项 附加资源许可证更改日志已知问题 描述 该示例 sampleUffSSD 预处理 TensorFlow …

js的入口函数有哪些?

在JavaScript中&#xff0c;常见的入口函数有以下几种&#xff1a; window.onload: 该函数会在页面完全加载完成后执行&#xff0c;包括所有图像、CSS样式表、JavaScript脚本等。document.ready: 该函数会在DOM&#xff08;文档对象模型&#xff09;完全加载完成后执行&#x…

pmp到底有多难考?

单说通过&#xff0c;难度不算特别高&#xff0c;但也有几点难度&#xff1a; 1、试卷难度&#xff1a;是笔试&#xff0c;180道题&#xff0c;题量大&#xff0c;比较机试耗时间&#xff1b; 2、题目难度&#xff1a;题目是中英文对照&#xff0c;可能有翻译不到位的地方&am…

0基础学习PyFlink——事件时间和运行时间的窗口

大纲 定制策略运行策略Reduce完整代码滑动窗口案例参考资料 在 《0基础学习PyFlink——时间滚动窗口(Tumbling Time Windows)》一文中&#xff0c;我们使用的是运行时间(Tumbling ProcessingTimeWindows)作为窗口的参考时间&#xff1a; reducedkeyed.window(TumblingProcess…

第4章_运算符

文章目录 1. 算术运算符1.1 加法与减法运算符1.2 乘法与除法运算符1.3 求模运算符 2. 比较运算符2.1 等号运算符2.2 安全等于运算符2.3 不等于运算符2.4 空运算符2.5 非空运算符2.6 最小值运算符2.7 最大值运算符2.8 BETWEEN AND运算符2.9 IN运算符2.10 NOT IN运算符2.11 LIKE运…

代码随想录算法训练营第三十九天丨 动态规划part02

62.不同路径 思路 动态规划 机器人从(0 , 0) 位置出发&#xff0c;到(m - 1, n - 1)终点。 按照动规五部曲来分析&#xff1a; 确定dp数组&#xff08;dp table&#xff09;以及下标的含义 dp[i][j] &#xff1a;表示从&#xff08;0 &#xff0c;0&#xff09;出发&#…

利用win32的GetLastInputInfo函数实现锁屏(C#)

前两天看到群里面讨论这个问题&#xff0c;刚好我们上一家公司的系统也有这个功能&#xff0c;就研究了一下&#xff0c;我们这边实现这个功能的目的如下&#xff1a;当用户长时间不操作系统时&#xff0c;自动退出系统并退回到登录界面&#xff0c;想要使用系统&#xff0c;就…

react官网

应急方案 – React 中文文档 (docschina.org) 正版卡死版 Hooks FAQ – React (reactjs.org) 英文流畅版 应急方案 – React 中文网 (nodejs.cn) 盗版流畅版&#xff08;翻译有稍稍的问题&#xff09; http://www.react-cn.com/index.html 黄版

【数据结构】数组和字符串(十三):链式字符串的基本操作(串长统计、查找、复制、插入、删除、串拼接)

文章目录 4.3 字符串4.3.1 字符串的定义与存储4.3.2 字符串的基本操作&#xff08;链式存储&#xff09;1. 结构体2. 初始化3. 判空4. 串尾添加5. 打印6. 串长统计7. 查找8. 复制9. 插入10. 删除11. 串拼接12. 销毁13. 主函数14. 代码整合 4.3 字符串 字符串(String)是由零个或…

Latex排版SIGGRAPH总结(持续总结中...)

本文学习总结自&#xff1a;How to use the ACM SIGGRAPH / TOG LaTeX template 相关文件&#xff1a;百度网盘 首先解压 “my paper” 中的文件&#xff0c;并用Latex打开mypaper.tex. 多行连等公式 \begin{equation}表示编号公式&#xff0c;\[ \]表示无编号公式 无编号\b…

go-zero数据库连接池 database/sql 源码学习

database/sql 中接口的层级关系https://draveness.me/golang/docs/part4-advanced/ch09-stdlib/golang-database-sql/ database/sql源码地址&#xff1a;https://github.com/golang/go/tree/release-branch.go1.17/src/database/sql go-zero数据库连接池源码地址 https://githu…

k8s中 RBAC中,clusterrole,serviceaccount , rolebinding 是什么关系谁先谁后

在Kubernetes的RBAC&#xff08;Role-Based Access Control&#xff09;中&#xff0c;ClusterRole、ServiceAccount和RoleBinding是三个关键的组件&#xff0c;它们之间的关系如下&#xff1a; ClusterRole&#xff1a;ClusterRole 是一种全局的权限规则&#xff0c;它定义了一…

ESP32网络开发实例-使用密码登录Web服务器

使用密码登录Web服务器 文章目录 使用密码登录Web服务器1、软件准备2、硬件准备3、代码实现在本文中,我们将使用 ESP32 和 Arduino IDE 设计一个受密码保护的 Web 服务器。 如果您使用 ESP32 制作了家庭自动化项目并且您正在访问 Web 服务器上的所有信息,并且您希望通过添加密…

JavaScript 中闭包是什么?有哪些应用场景?

给大家推荐一个实用面试题库 1、前端面试题库 &#xff08;面试必备&#xff09; 推荐&#xff1a;★★★★★ 地址&#xff1a;web前端面试题库 闭包是什么&#xff1f; 闭包是指一个函数可以访问并操作其词法作用域外的变量的能力。闭包就是能够读取其他函数内…

双轮差速模型机器人通过线速度、角速度计算机器人位姿

已知上一时刻机器人位置P_OLD (x,y,),机器人当前时刻的线速度和角速度&#xff08;v,&#xff09;,短时间内t内&#xff0c;机器人在线性部分和非线性部分的增量为 线性部分&#xff1a; 非线性部分&#xff1a; 由于可能非常小&#xff0c;导致非线性部分数值不稳定&#xf…

【R统计】各式各样的插补法解决数据缺失的问题!

&#x1f482; 个人信息&#xff1a;酷在前行&#x1f44d; 版权: 博文由【酷在前行】原创、需要转载请联系博主&#x1f440; 如果博文对您有帮助&#xff0c;欢迎点赞、关注、收藏 订阅专栏&#x1f516; 本文收录于【R统计】&#xff0c;该专栏主要介绍R语言实现统计分析的…