数据结构和算法学习笔记(三)二叉树
目录
警告
本文最后更新于 2022-09-13,文中内容可能已过时。
1. 遍历二叉树
递归序
1,2,4,4,4,2,5,5,5,2,1,3,6,6,6,3,7,7,7,3,1
即为递归序
先序遍历
- 对于每棵子树,先打印头节点,再打印左子树,再打印右子树
- 在递归序中,只有第一次碰到,才打印
中序遍历
- 对于每棵子树,先打印左子树,再打印头节点,再打印右子树
- 在递归序中,只有第二次碰到,才打印
后序遍历
- 对于每棵子树,先打印头节点,再打印左子树,再打印右子树
- 在递归序中,只有第三次碰到,才打印
非递归实现
任何递归都可以改为非递归:自己压栈不就得了
先序遍历
- 准备一个栈
- 头节点压栈
- 弹出一个节点,打印
- 先压右节点,再压左节点
- 回到第
3
步
中序遍历
- 准备一个栈
- 从头节点开始,依次把所有左边界(一路顺着
left
走,直到没有left
)节点压栈(包括头节点) - 弹出一个节点,打印
- 弹出的节点如果有右子树,回到第
2
步
后序遍历
- 准备两个栈
s1
,s2
- 头节点压
s1
- 弹出一个节点,放入
s2
s1
先压左节点,再压右节点- 回到第
3
步 - 循环完成后,依次弹出
s2
打印
- 准备两个栈
算法题
- 递归和非递归的实现二叉树的先序、中序、后序遍历
- 655. 输出二叉树
|
|
- 102. 二叉树的层序遍历
- 广度优先遍历:
- 头节点入队列
- 弹出节点,打印
- 左子节点入队列,右子节点入队列
- 回到步骤
1
- 广度优先遍历:
- 662. 二叉树最大宽度
- 注意:不能硬把每层的节点(包括空节点)记下来,然后去数,否则多层之后,中间空节点可能成
2 ^ cur_depth
指数级增加,导致超时 - 方法:要记录每个节点的
postion
,把空间的算上,是一颗完全二叉树,可根据堆排序知识点算position
,用 position 相减,避免指数级空节点
- 注意:不能硬把每层的节点(包括空节点)记下来,然后去数,否则多层之后,中间空节点可能成
2. 二叉树的相关概念及判断
2.1 二叉搜索树
若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值; 若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值
判定:
二叉搜索树的中序遍历是递增的
按定义:
- 左子树是搜索二叉树
- 右子树是搜索二叉树
left_max < node.val < right.min
算法题:
2.2 完全二叉树
- 叶子结点只能出现在最下层和次下层,且最下层的叶子结点集中在树的左部。
- 即:最后一层可能不满,但左边一定是连续的
判定:
广度优先搜索:
- 任一节点如果有右孩子,没有左孩子,返回
False
条件 1
不触发的前提下,遇到第一个左右孩子不全的节点,后面必须全是叶子节点,否则返回False
- 任一节点如果有右孩子,没有左孩子,返回
数索引:
- 按堆排序类似的索引法:
idx_left == idx * 2 + 1
,idx_right == idx * 2 + 2
给每个节点定索引 - 最后一个节点的索引等于节点总数加 1,否则返回
False
- 按堆排序类似的索引法:
算法题
2.3 满二叉树
- 国际定义:一棵二叉树的节点要么是叶子结点,要么它有两个子节点
- 国内定义:每层的结点数都达到最大值(国际定义为:Perfect binary tree)
2.4 平衡二叉树 (AVL树)
左子树和右子树的高度之差(平衡因子)的绝对值不超过1且它的左子树和右子树都是一颗平衡二叉树
判定:
满足三个条件:
- 左子树是平衡二叉树
- 右子树是平衡二叉树
- 左右子树高度差不超过 1
求高度和判定当前是否为平衡二叉树,可在一个函数中完成:函数返回两个值
<bool:is_avl, int:height>
即可
算法题:
2.5 解题套路
问子树要信息(分解成子问题,如上述平衡二叉树判定),可解决树型DP问题
3. 二叉树算法题
3.1 最近公共祖先
3.2 前驱节点和后继结点
3.3 二叉树序列化和反序列化
- 297. 二叉树的序列化与反序列化
- 先序遍历 (With null node)
- 层次遍历 (With null node)
- 先序 + 中序 重建二叉树
3.4 折纸问题
请把纸条竖着放在桌⼦上,然后从纸条的下边向上⽅对折,压出折痕后再展 开。此时有1条折痕,突起的⽅向指向纸条的背⾯,这条折痕叫做“下”折痕 ;突起的⽅向指向纸条正⾯的折痕叫做“上”折痕。如果每次都从下边向上⽅ 对折,对折N次。请从上到下计算出所有折痕的⽅向。
给定折的次数n,请返回从上到下的折痕的数组,若为下折痕则对应元素为"down",若为上折痕则为"up".
解法:举例、模拟得出数学模型:一颗满二叉树,各个节点直接有规律,中序遍历得结果
注意:并不需要真正构建二叉树(空间复杂度: $O(N)$,N 为树得深度),而构建一个二叉树需要 $O(2 ^ N)$ 的空间
golang
代码:
|
|