LeetCodeCampsDay16
LeetCodeCampsDay16
记录下,今儿三个题目一次过;
其中路径总和问题和之前求二叉树的所有路径相似
找树左下角的值可以用层序解决
从中序与后序遍历序列构造二叉树 可以用递归方法,找到输入输出;终止条件;单层逻辑就能解决
找树左下角的值
https://leetcode.cn/problems/find-bottom-left-tree-value/
给定一个二叉树的 根节点 root
,请找出该二叉树的 最底层 最左边 节点的值。
假设二叉树中至少有一个节点。
示例 1:
1 | 输入: root = [2,1,3] |
示例 2:
1 | 输入: [1,2,3,4,null,5,6,null,null,7] |
提示:
- 二叉树的节点个数的范围是
[1,104]
-231 <= Node.val <= 231 - 1
层序迭代思路
- 使用迭代–层序的方法, 令res存储每层第一个元素,遍历到最后一层结束,再返回res即可
层序迭代代码
- 时间复杂度O(N)
- 空间复杂度O(W)—二叉树最大宽度(也即二叉树每层最大长度)
1 | # Definition for a binary tree node. |
112. 路径总和
https://leetcode.cn/problems/path-sum/
给你二叉树的根节点 root
和一个表示目标和的整数 targetSum
。判断该树中是否存在 根节点到叶子节点 的路径,这条路径上所有节点值相加等于目标和 targetSum
。如果存在,返回 true
;否则,返回 false
。
叶子节点 是指没有子节点的节点。
示例 1:
1 | 输入:root = [5,4,8,11,null,13,4,7,2,null,null,null,1], targetSum = 22 |
示例 2:
1 | 输入:root = [1,2,3], targetSum = 5 |
示例 3:
1 | 输入:root = [], targetSum = 0 |
提示:
- 树中节点的数目在范围
[0, 5000]
内 -1000 <= Node.val <= 1000
-1000 <= targetSum <= 1000
后序迭代思路
- 这题目和“二叉树所有路径”(day15)相似,只是将保存路径变成
保存路径上所有元素的和
;遇到叶子节点时,判断这和是否与target相等即可 - 本题同样对前/中/后,甚至逆向的前/中/后顺序不敏感,所有顺序都能通过
- 下面代码使用后序
后序迭代代码
- 时间复杂度O(N)
- 空间复杂度O(H)
1 | # Definition for a binary tree node. |
递归思路
- 输入输出:输入节点和targetSum,这里的targetSum每次都需要变小; 输出True/False
- 终止条件:不能是判断node是否为空,因为它说明不了什么;需要判断是否为叶子节点,并且判断targetSum是否为零,为零则返回True;如果叶子节点且targetSum不为零,返回False
- 单层逻辑,判断左、右子树是否已经存在了满足target的路径,存在就返回True
递归代码
- 时间复杂度O(N)
- 空间复杂度O(N)
1 | # Definition for a binary tree node. |
113. 路径总和 II
https://leetcode.cn/problems/path-sum-ii/
给你二叉树的根节点 root
和一个整数目标和 targetSum
,找出所有 从根节点到叶子节点 路径总和等于给定目标和的路径。
叶子节点 是指没有子节点的节点。
示例 1:
1 | 输入:root = [5,4,8,11,null,13,4,7,2,null,null,5,1], targetSum = 22 |
示例 2:
1 | 输入:root = [1,2,3], targetSum = 5 |
示例 3:
1 | 输入:root = [1,2], targetSum = 0 |
提示:
- 树中节点总数在范围
[0, 5000]
内 -1000 <= Node.val <= 1000
-1000 <= targetSum <= 1000
思路
- 和之前做的差不多,但注意,这里是将
pathNode + [node.right.val]
传进去path里,所以对pathNode本身没有修改,如果直接修改pathNode,会把所有路径的值都添加进去,就不对了
迭代代码
- 时间复杂度O(N^2)
- 空间复杂度O(N^2)
1 | # Definition for a binary tree node. |
递归代码
- 时间复杂度O(N)
- 空间复杂度O(N)
1 | # Definition for a binary tree node. |
106. 从中序与后序遍历序列构造二叉树
https://leetcode.cn/problems/construct-binary-tree-from-inorder-and-postorder-traversal/
给定两个整数数组 inorder
和 postorder
,其中 inorder
是二叉树的中序遍历, postorder
是同一棵树的后序遍历,请你构造并返回这颗 二叉树 。
示例 1:
1 | 输入:inorder = [9,3,15,20,7], postorder = [9,15,7,20,3] |
示例 2:
1 | 输入:inorder = [-1], postorder = [-1] |
提示:
1 <= inorder.length <= 3000
postorder.length == inorder.length
-3000 <= inorder[i], postorder[i] <= 3000
inorder
和postorder
都由 不同 的值组成postorder
中每一个值都在inorder
中inorder
保证是树的中序遍历postorder
保证是树的后序遍历
递归思路
- 用两个指针,一个指针指向inorder首,一个指向postorder末尾
- 使用while循环,当left.val == right.val时
- 右子树
- left的下标也是右子树的postorder的开始下标
- right - 1的下标是右子树postorder的结束下标(注意python里列表的右是开区间,所以写postorder[left: right]即可)
- left + 1 是inorder中右子树开始下标,并且后面全是右子树
- 左子树
- left 是inorder中左子树结束下标,并且前面全是左子树
- left 是postorder中左子树结束下标,并且前面全是左子树
把左子树想成一大块儿;右子树想成一大块儿,就好理解了
in | 左子树 |
左子树 |
3 | 右子树 | 右子树 | 右子树 |
Post | 左子树 |
左子树 |
右子树 | 右子树 | 右子树 | 3 |
left | Right |
这是对于3的两个子树的区间位置
1 | leftTree = self.foo(inorder[:left], postorder[:left]) |
递归代码
- 时间复杂度O(N^2)—最好情况O(NlogN)
- 空间复杂度O(N^2)
- 可以将while找root替换成index函数,即
left = inorder.index(postorder[right])
1 | # Definition for a binary tree node. |
105. 从前序与中序遍历序列构造二叉树
https://leetcode.cn/problems/construct-binary-tree-from-preorder-and-inorder-traversal/
给定两个整数数组 preorder
和 inorder
,其中 preorder
是二叉树的先序遍历, inorder
是同一棵树的中序遍历,请构造二叉树并返回其根节点。
示例 1:
1 | 输入: preorder = [3,9,20,15,7], inorder = [9,3,15,20,7] |
示例 2:
1 | 输入: preorder = [-1], inorder = [-1] |
提示:
1 <= preorder.length <= 3000
inorder.length == preorder.length
-3000 <= preorder[i], inorder[i] <= 3000
preorder
和inorder
均 无重复 元素inorder
均出现在preorder
preorder
保证 为二叉树的前序遍历序列inorder
保证 为二叉树的中序遍历序列
代码
- 时间复杂度O(N)—最好情况O(logN)
- 空间复杂度O(N)
1 | # Definition for a binary tree node. |
889. 根据前序和后序遍历构造二叉树
https://leetcode.cn/problems/construct-binary-tree-from-preorder-and-postorder-traversal/
给定两个整数数组,preorder
和 postorder
,其中 preorder
是一个具有 无重复 值的二叉树的前序遍历,postorder
是同一棵树的后序遍历,重构并返回二叉树。
如果存在多个答案,您可以返回其中 任何 一个。
示例 1:
1 | 输入:preorder = [1,2,4,5,3,6,7], postorder = [4,5,2,6,7,3,1] |
示例 2:
1 | 输入: preorder = [1], postorder = [1] |
提示:
1 <= preorder.length <= 30
1 <= preorder[i] <= preorder.length
preorder
中所有值都 不同postorder.length == preorder.length
1 <= postorder[i] <= postorder.length
postorder
中所有值都 不同- 保证
preorder
和postorder
是同一棵二叉树的前序遍历和后序遍历
递归思路
- 和中充&后序构造/前序&中序构造不太一样,只知道前序和后序,需要更精确地知道左、右子树的区间;
- 除了需要知道当前root节点rootIndexPost以外,还需要知道左(或者右)的root节点rootIndexLeftTreeInPost的位置,这样才能判断左(或右)子树的区间
- 输入输出:输入为前&后序列,输出为子树
- 终止条件:序列为空则返回None表示空节点,序列长度为1则返回这个叶子节点
- 单层逻辑:找到root(preorder的第一个就是),再以当前root为中心,找到它的左、右子树区间;使用递归找左、右子树,按拼接到root上就好;难度在于找到左、右树区间范围
递归代码
1 | # Definition for a binary tree node. |