LeetCode 102 二叉树的层序遍历
题目链接:https://leetcode.cn/problems/binary-tree-level-order-traversal/
题目讲解:https://www.bilibili.com/video/BV1GY4y1u7b2/
题目描述:给你二叉树的根节点 root ,返回其节点值的 层序遍历 。 (即逐层地,从左到右访问所有节点)。
示例1:输入:root = [3,9,20,null,null,15,7]
输出:[[3],[9,20],[15,7]]
示例2:输入:root = [1]
输出:[[1]]
示例3:输入:root = []
输出:[]
思路:层序遍历一个二叉树。就是从左到右一层一层的去遍历二叉树。需要借用一个辅助数据结构即队列来实现,队列先进先出,符合一层一层遍历的逻辑,而用栈先进后出适合模拟深度优先遍历也就是递归的逻辑。而这种层序遍历方式就是图论中的广度优先遍历,只不过我们应用在二叉树上。
class Solution {
public List<List<Integer>> resList = new ArrayList<List<Integer>>();//在List中存放List:
public List<List<Integer>> levelOrder(TreeNode root) {
if(root == null) return resList;
Queue<TreeNode> que = new LinkedList<TreeNode>();
que.offer(root);//添加一个元素并返回true
while(!que.isEmpty()){
List<Integer> itemList = new ArrayList<Integer>();
int len = que.size();
while(len-- > 0){
TreeNode tmpNode = que.poll();//移除并返问队列头部的元素
itemList.add(tmpNode.val);
if(tmpNode.left != null) que.offer(tmpNode.left);
if(tmpNode.right != null) que.offer(tmpNode.right);
}
resList.add(itemList);
}
return resList;
}
}
/**
* Definition for a binary tree node.
* public class TreeNode {
* int val;
* TreeNode left;
* TreeNode right;
* TreeNode() {}
* TreeNode(int val) { this.val = val; }
* TreeNode(int val, TreeNode left, TreeNode right) {
* this.val = val;
* this.left = left;
* this.right = right;
* }
* }
*/
Java 队列 queue的一些操作:
- add 增加一个元索 如果队列已满,则抛出一个IIIegaISlabEepeplian异常
- remove 移除并返回队列头部的元素 如果队列为空,则抛出一个NoSuchElementException异常
- element 返回队列头部的元素 如果队列为空,则抛出一个NoSuchElementException异常
- offer 添加一个元素并返回true 如果队列已满,则返回false
- poll 移除并返问队列头部的元素 如果队列为空,则返回null
- peek 返回队列头部的元素 如果队列为空,则返回null
- put 添加一个元素 如果队列满,则阻塞
- take 移除并返回队列头部的元素 如果队列为空,则阻塞
LeetCode 107 二叉树的层序遍历 II
题目链接:https://leetcode.cn/problems/binary-tree-level-order-traversal-ii/
题目描述:给你二叉树的根节点 root ,返回其节点值 自底向上的层序遍历 。 (即按从叶子节点所在层到根节点所在的层,逐层从左向右遍历)
示例1:输入:root = [3,9,20,null,null,15,7]
输出:[[3],[9,20],[15,7]]
示例2:输入:root = [1]
输出:[[1]]
示例3:输入:root = []
输出:[]
思路:相对于102.二叉树的层序遍历,就是最后把result数组反转一下就可以了。
class Solution {
public List<List<Integer>> levelOrderBottom(TreeNode root) {
List<List<Integer>> list = new ArrayList<>();
Deque<TreeNode> que = new LinkedList<>();
if(root == null) return list;
que.offerLast(root);
while(!que.isEmpty()){
List<Integer> levelList = new ArrayList<>();
int levelSize = que.size();
for(int i = 0; i < levelSize; i++){
TreeNode peek = que.peekFirst();
levelList.add(que.pollFirst().val);
if(peek.left != null) que.offerLast(peek.left);
if(peek.right != null) que.offerLast(peek.right);
}
list.add(levelList);
}
List<List<Integer>> result = new ArrayList<>();
for(int i = list.size() - 1; i >= 0; i--){
result.add(list.get(i));
}
return result;
}
}
/**
* Definition for a binary tree node.
* public class TreeNode {
* int val;
* TreeNode left;
* TreeNode right;
* TreeNode() {}
* TreeNode(int val) { this.val = val; }
* TreeNode(int val, TreeNode left, TreeNode right) {
* this.val = val;
* this.left = left;
* this.right = right;
* }
* }
*/
199 二叉树的右视图
题目链接:https://leetcode.cn/problems/binary-tree-right-side-view/
题目描述:给定一个二叉树的 根节点 root,想象自己站在它的右侧,按照从顶部到底部的顺序,返回从右侧所能看到的节点值。
示例1:输入: [1,2,3,null,5,null,4]
输出: [1,3,4]
示例2:输入: [1,null,3]
输出: [1,3]
示例3:输入: []
输出: []
思路:层序遍历的时候,判断是否遍历到单层的最后面的元素,如果是,就放进result数组中,随后返回result就可以了。
class Solution {
public List<Integer> rightSideView(TreeNode root) {
List<Integer> list = new ArrayList<>();
Deque<TreeNode> que = new LinkedList<>();
if(root == null) return list;
que.offerLast(root);
while(!que.isEmpty()){
int levelSize = que.size();
for(int i = 0; i < levelSize; i++){
TreeNode poll = que.pollFirst();
if(poll.left != null) que.addLast(poll.left);
if(poll.right != null) que.addLast(poll.right);
if(i == levelSize - 1) list.add(poll.val);
}
}
return list;
}
}
637 二叉树的层平均值
题目链接:https://leetcode.cn/problems/average-of-levels-in-binary-tree/
题目描述:给定一个非空二叉树的根节点 root , 以数组的形式返回每一层节点的平均值。与实际答案相差 10-5 以内的答案可以被接受。
示例1:输入:root = [3,9,20,null,null,15,7]
输出:[3.00000,14.50000,11.00000]
解释:第 0 层的平均值为 3,第 1 层的平均值为 14.5,第 2 层的平均值为 11 。
因此返回 [3, 14.5, 11] 。
示例2:输入:root = [3,9,20,15,7]
输出:[3.00000,14.50000,11.00000]
思路:本题就是层序遍历的时候把一层求个总和再取一个均值。
class Solution {
public List<Double> averageOfLevels(TreeNode root) {
List<Double> list = new ArrayList<>();
Deque<TreeNode> que = new LinkedList<>();
if(root == null){
return list;
}
que.offerLast(root);
while(!que.isEmpty()){
int levelSize = que.size();
double levelSum = 0.0;
for(int i = 0; i < levelSize; i++){
TreeNode poll = que.pollFirst();
levelSum += poll.val;
if(poll.left != null){
que.addLast(poll.left);
}
if(poll.right != null){
que.addLast(poll.right);
}
}
list.add(levelSum / levelSize);
}
return list;
}
}
429 N叉树的层序遍历
题目链接:https://leetcode.cn/problems/n-ary-tree-level-order-traversal/
题目描述:给定一个 N 叉树,返回其节点值的层序遍历。(即从左到右,逐层遍历)。树的序列化输入是用层序遍历,每组子节点都由 null 值分隔(参见示例)。
示例1:
输入:root = [1,null,3,2,4,null,5,6]
输出:[[1],[3,2,4],[5,6]]
示例2:
输入:root = [1,null,2,3,4,5,null,null,6,7,null,8,null,9,10,null,null,11,null,12,null,13,null,null,14]
输出:[[1],[2,3,4,5],[6,7,8,9,10],[11,12,13],[14]]
class Solution {
public List<List<Integer>> levelOrder(Node root) {
List<List<Integer>> list = new ArrayList<>();
Deque<Node> que = new LinkedList<>();
if(root == null){
return list;
}
que.offerLast(root);
while(!que.isEmpty()){
int levelSize = que.size();
List<Integer> levelList = new ArrayList<>();
for(int i = 0; i < levelSize; i++){
Node poll = que.pollFirst();
levelList.add(poll.val);
List<Node> children = poll.children;
if(children == null || children.size() == 0){
continue;
}
for(Node child : children){
if(child != null){
que.offerLast(child);
}
}
}
list.add(levelList);
}
return list;
}
}
515.在每个树行中找最大值
题目链接:https://leetcode.cn/problems/find-largest-value-in-each-tree-row/
题目描述:给定一棵二叉树的根节点 root ,请找出该二叉树中每一层的最大值。
示例1:输入: root = [1,3,2,5,3,null,9]
输出: [1,3,9]
示例2:输入: root = [1,2,3]
输出: [1,3]
class Solution {
public List<Integer> largestValues(TreeNode root) {
if(root == null){
return Collections.emptyList();//主要目的就是返回一个不可变的列表,使用这个方法作为返回值就不需要再创建一个新对象,可以减少内存开销。并且返回一个size为0的List,调用者不需要校验返回值是否为null,所以建议使用这个方法返回可能为空的List。
}
List<Integer> result = new ArrayList();
Queue<TreeNode> queue = new LinkedList();
queue.offer(root);
while(!queue.isEmpty()){
int max = Integer.MIN_VALUE;
for(int i = queue.size(); i > 0; i--){
TreeNode node = queue.poll();
max = Math.max(max, node.val);
if(node.left != null) queue.offer(node.left);
if(node.right != null) queue.offer(node.right);
}
result.add(max);
}
return result;
}
}
116.填充每个节点的下一个右侧节点指针
题目链接:https://leetcode.cn/problems/populating-next-right-pointers-in-each-node/
题目描述:给定一个 完美二叉树 ,其所有叶子节点都在同一层,每个父节点都有两个子节点。二叉树定义如下:
struct Node {
int val;
Node *left;
Node *right;
Node *next;
}
填充它的每个 next 指针,让这个指针指向其下一个右侧节点。如果找不到下一个右侧节点,则将 next 指针设置为 NULL。
初始状态下,所有 next 指针都被设置为 NULL。
示例1:输入:root = [1,2,3,4,5,6,7]
输出:[1,#,2,3,#,4,5,6,7,#]
解释:给定二叉树如图 A 所示,你的函数应该填充它的每个 next 指针,以指向其下一个右侧节点,如图 B 所示。序列化的输出按层序遍历排列,同一层节点由 next 指针连接,‘#’ 标志着每一层的结束。
示例2:输入:root = []
输出:[]
思路:本题依然是层序遍历,只不过在单层遍历的时候记录一下本层的头部节点,然后在遍历的时候让前一个节点指向本节点就可以了
class Solution {
public Node connect(Node root) {
Queue<Node> tmpQueue = new LinkedList<Node>();
if(root != null) tmpQueue.add(root);
while(tmpQueue.size() != 0){
int size = tmpQueue.size();
Node cur = tmpQueue.poll();
if(cur.left != null) tmpQueue.add(cur.left);
if(cur.right != null) tmpQueue.add(cur.right);
for(int index = 1; index < size; index++){
Node next = tmpQueue.poll();
if(next.left != null) tmpQueue.add(next.left);
if(next.right != null) tmpQueue.add(next.right);
cur.next = next;
cur = next;
}
}
return root;
}
}
117.填充每个节点的下一个右侧节点指针II
题目链接:https://leetcode.cn/problems/populating-next-right-pointers-in-each-node-ii/
题目描述:给定一个二叉树:
struct Node {
int val;
Node *left;
Node *right;
Node *next;
}
填充它的每个 next 指针,让这个指针指向其下一个右侧节点。如果找不到下一个右侧节点,则将 next 指针设置为 NULL 。
初始状态下,所有 next 指针都被设置为 NULL 。
示例1:输入:root = [1,2,3,4,5,null,7]
输出:[1,#,2,3,#,4,5,7,#]
解释:给定二叉树如图 A 所示,你的函数应该填充它的每个 next 指针,以指向其下一个右侧节点,如图 B 所示。序列化输出按层序遍历顺序(由 next 指针连接),‘#’ 表示每层的末尾。
示例2:输入:root = []
输出:[]
class Solution {
public Node connect(Node root) {
Queue<Node> queue = new LinkedList<>();
if(root != null) queue.add(root);
while(!queue.isEmpty()){
int size = queue.size();
Node node = null;
Node nodepre = null;
for(int i = 0; i < size; i++){
if(i == 0){
nodepre = queue.poll();// 取出本层头一个节点
node = nodepre;
}else{
node = queue.poll();
nodepre.next = node;// 本层前一个节点 next 指向当前节点
nodepre = nodepre.next;
}
if (node.left != null) {
queue.add(node.left);
}
if (node.right != null) {
queue.add(node.right);
}
}
nodepre.next = null;// 本层最后一个节点 next 指向 null
}
return root;
}
}
104.二叉树的最大深度
题目链接:https://leetcode.cn/problems/maximum-depth-of-binary-tree/
题目描述:给定一个二叉树,找出其最大深度。
二叉树的深度为根节点到最远叶子节点的最长路径上的节点数。
说明: 叶子节点是指没有子节点的节点。
示例:给定二叉树 [3,9,20,null,null,15,7],返回它的最大深度 3 。
class Solution {
public int maxDepth(TreeNode root) {
if(root == null) return 0;
Queue<TreeNode> que = new LinkedList<>();
que.offer(root);
int depth = 0;
while(!que.isEmpty()){
int len = que.size();
while(len > 0){
TreeNode node = que.poll();
if(node.left != null) que.offer(node.left);
if(node.right != null) que.offer(node.right);
len--;
}
depth++;
}
return depth;
}
}
111.二叉树的最小深度
题目链接:https://leetcode.cn/problems/minimum-depth-of-binary-tree/
题目描述:给定一个二叉树,找出其最小深度。
最小深度是从根节点到最近叶子节点的最短路径上的节点数量。
说明:叶子节点是指没有子节点的节点。
示例1:输入:root = [3,9,20,null,null,15,7]
输出:2
示例2:输入:root = [2,null,3,null,4,null,5,null,6]
输出:5
class Solution {
public int minDepth(TreeNode root) {
if(root == null) return 0;
Queue<TreeNode> que = new LinkedList<>();
que.offer(root);
int depth = 0;
while(!que.isEmpty()){
int size = que.size();
depth++;
TreeNode cur = null;
for(int i = 0; i < size; i++){
cur = que.poll();
if(cur.left == null && cur.right == null){
return depth;
}
if(cur.left != null) que.offer(cur.left);
if(cur.right != null) que.offer(cur.right);
}
}
return depth;
}
}
226. 翻转二叉树
题目链接:https://leetcode.cn/problems/invert-binary-tree/
题目描述:给你一棵二叉树的根节点 root ,翻转这棵二叉树,并返回其根节点。
示例1:输入:root = [4,2,7,1,3,6,9]
输出:[4,7,2,9,6,3,1]
示例2:输入:root = [2,1,3]
输出:[2,3,1]
示例3:输入:root = []
输出:[]
class Solution {
/**
* 前后序遍历都可以
* 中序不行,因为先左孩子交换孩子,再根交换孩子(做完后,右孩子已经变成了原来的左孩子),再右孩子交换孩子(此时其实是对原来的左孩子做交换)
*/
public TreeNode invertTree(TreeNode root) {
if(root == null){
return null;
}
invertTree(root.left);
invertTree(root.right);
swapchildren(root);
return root;
}
private void swapchildren(TreeNode root){
TreeNode tmp = root.left;
root.left = root.right;
root.right = tmp;
}
}
101. 对称二叉树
题目链接:https://leetcode.cn/problems/symmetric-tree/
题目讲解:https://www.bilibili.com/video/BV1ue4y1Y7Mf/
题目描述:给你一个二叉树的根节点 root , 检查它是否轴对称。
示例1:输入:root = [1,2,2,3,4,4,3]
输出:true
示例2:输入:root = [1,2,2,null,3,null,3]
输出:false
思路:本题遍历只能是“后序遍历”,因为我们要通过递归函数的返回值来判断两个子树的内侧节点和外侧节点是否相等。正是因为要遍历两棵树而且要比较内侧和外侧节点,所以准确的来说是一个树的遍历顺序是左右中,一个树的遍历顺序是右左中。但都可以理解算是后序遍历,尽管已经不是严格上在一个树上进行遍历的后序遍历了。
class Solution {
public boolean isSymmetric(TreeNode root) {
return compare(root.left, root.right);
}
private boolean compare(TreeNode left, TreeNode right){
if(left == null && right != null){
return false;
}
if(left != null && right == null){
return false;
}
if(left == null && right == null){
return true;
}
if(left.val != right.val){
return false;
}
boolean compareOutside = compare(left.left, right.right);
boolean compareInside = compare(left.right, right.left);
return compareOutside && compareInside;
}
}
总结:昨天的任务到今天才写完,不怎么会写,还有多学学啊!!