算法打卡day13|二叉树篇02|Leetcode 102.二叉树的层序遍历、226.翻转二叉树、101. 对称二叉树

  在做题之前说明

DequeQueue

在Java中,DequeQueue是两种不同的数据结构接口,它们都继承自Collection接口;DequeQueue的超集,提供了更多的操作和灵活性,以下它们之间存在一些关键的区别:

  1. 操作范围

    • Queue接口仅定义了一端(队尾)用于插入元素,另一端(队首)用于删除元素的操作,这符合先进先出(FIFO)的原则。
    • Deque接口则扩展了Queue的能力,允许在双端进行插入和删除元素,因此它不仅支持FIFO原则,还可以支持后进先出(LIFO)原则,类似于栈。
  2. 具体实现

    • Queue接口的典型实现是LinkedListPriorityQueue。其中,LinkedList实现了所有Queue接口的方法,同时还支持Deque接口的方法。而PriorityQueue则是一个基于优先级堆的无界队列,它不支持Deque接口。
    • Deque接口的实现包括ArrayDequeLinkedListArrayDeque是基于数组的双端队列实现,通常比LinkedList更高效,特别是在随机访问和尾部操作方面。LinkedList也实现了Deque接口,因此它既可以作为队列使用,也可以作为双端队列使用。
  3. 方法差异

    • Queue接口提供的方法如offer(E e)用于添加元素,poll()用于移除并返回队首元素,peek()用于查看队首元素而不移除。
    • Deque接口除了包含Queue的所有方法外,还提供了额外的方法,如addFirst(E e)addLast(E e)用于在双端分别添加元素,removeFirst()removeLast()用于从双端移除元素,以及getFirst()getLast()用于获取双端的元素而不移除。
  4. 性能考虑

    • 对于需要频繁在两端插入和删除元素的场景,Deque通常是更好的选择,因为它专门为此设计。
    • 如果应用场景只需要队列的基本操作(即只有一端添加,另一端删除),那么使用Queue可能更为简洁,尤其是当使用PriorityQueue实现时,它提供了特定的排序机制。

offer()方法与offerLast()

Deque接口中的offer()方法与offerLast()方法都用于向双端队列的末尾添加元素,但它们之间有细微的差别。

offer()方法Queue接口中定义的,它被所有实现了Queue接口的类继承,包括Deque。当你调用offer()方法时,它会将元素添加到Deque的末尾。如果Deque已经满了offer()方法将抛出一个IllegalStateException。而offerLast()方法是Deque接口特有的,Deque已满时不会抛出异常,而是返回一个特殊的值false,表明元素没有被添加到队列中。这种行为使得offerLast()方法在某些情况下更加健壮,因为它避免了因队列满而导致的程序异常。

peek()peekFirst()

peek()方法是从Queue接口继承来的,它返回队列头部的元素,但不移除它。如果队列为空,则peek()方法返回null

peekFirst()方法是Deque接口特有,它与peek()方法类似,也是返回队列头部的元素,但不移除它。然而,与peek()方法不同的是,peekFirst()方法不会抛出空指针异常,即使队列为空。当队列为空时,peekFirst()方法返回null。这使得peekFirst()方法在某些情况下更加安全和可靠

以此类推,其他的比如poll()和pollFirst()也是如此;

算法题

Leetcode  102.二叉树的层序遍历

题目链接:102.二叉树的层序遍历

大佬视频讲解:二叉树的层序遍历视频讲解

个人思路

层序遍历也就是图论中的广度优先遍历,是使用队列的结构;用队列一层层遍历,并加入结果数组,最后返回;

解法
迭代法
class Solution {public List<List<Integer>> resList = new ArrayList<List<Integer>>();public List<List<Integer>> levelOrder(TreeNode root) {checkFun01(root);return resList;}public void checkFun01(TreeNode root){if(root==null) return;Queue<TreeNode> que=new LinkedList<TreeNode>();que.offer(root);//加入节点while(!que.isEmpty()){//终止条件int len =que.size();//当前层数 元素的数量List<Integer> itemList=new ArrayList<Integer>();while(len>0){TreeNode temp=que.poll();itemList.add(temp.val);//结果子列表加入值if(temp.left!=null) que.offer(temp.left);//加入左节点if(temp.right!=null) que.offer(temp.right);len--;//遍历下一个节点}resList.add(itemList);//结果列表加入子列表}}}

时间复杂度:O(n);(遍历整棵树)

空间复杂度:O(n);(使用一个结果子列表,和一个列表)

递归法
class Solution {//1.确定递归函数的参数和返回值public List<List<Integer>> resList = new ArrayList<List<Integer>>();public List<List<Integer>> levelOrder(TreeNode root) {checkFun02(root,0);return resList;}public void checkFun02(TreeNode root, int deep){if(root==null) return;//2.确定递归终止条件//3.确定单层递归的逻辑deep++;//层级if(resList.size()<deep){//利用list的索引值进行层级界定List<Integer> item=new ArrayList<Integer>();resList.add(item); //当层级增加时,list的Item也增加}resList.get(deep-1).add(root.val);//get 层级 checkFun02(root.left,deep);//递归左子树checkFun02(root.right,deep);//递归右子树}
}

时间复杂度:O(n);(遍历整棵树)

空间复杂度:O(n);(使用一个结果子列表,和一个列表)


Leetcode 226.翻转二叉树

题目链接:226.翻转二叉树

大佬视频讲解:翻转二叉树视频讲解

个人思路

遍历二叉树,将每个节点的左右孩子交换一下

解法

这道题关键在于遍历顺序,这道题目使用前序遍历,后序遍历和层序遍历都可以,唯独中序遍历不方便,因为中序遍历会把某些节点的左右孩子翻转了两次

递归法(DFS)
class Solution {public TreeNode invertTree(TreeNode root) {//1.确定递归函数的参数和返回值if (root == null) {//2.确定终止条件return null;}//3.确定单层递归的逻辑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;}
}

时间复杂度:O(n);(遍历整棵树)

空间复杂度:O(n);(递归树的高度h,使用临时节点来换位)

迭代法(BFS)

class Solution {public TreeNode invertTree(TreeNode root) {if (root == null) {return null;}ArrayDeque<TreeNode> deque = new ArrayDeque<>();//队列进行层序遍历deque.offer(root);while (!deque.isEmpty()) {int size = deque.size();while (size-- > 0) {//遍历各层的各个节点TreeNode node = deque.poll();swap(node);//交换节点的左右子树if (node.left != null) deque.offer(node.left);if (node.right != null) deque.offer(node.right);}}return root;}public void swap(TreeNode root) {TreeNode temp = root.left;root.left = root.right;root.right = temp;}
}

时间复杂度:O(n);(遍历整棵树)

空间复杂度:O(n);(使用队列进行层序遍历和临时节点来换位)


Leetcode 101. 对称二叉树

题目链接:101. 对称二叉树

大佬视频讲解:对称二叉树视频讲解

个人思路

将二叉树分成两半一边为左子树A,一边为右子树B,然后就对比 A节点的左边和B节点的右边,A节点的右边和B节点的左边,都相等就对称了

解法
递归法

因为要遍历两棵树而且要比较内侧和外侧节点,所以一个树的遍历顺序是左右中,一个树的遍历顺序是右左中

递归三部曲

1.确定递归函数的参数和返回值

因为要比较的是根节点的两个子树是否是相互翻转的,参数就是左子树节点和右子树节点。返回值为布尔类型。

2.确定终止条件

要比较两个节点数值相不相同,要把两个节点为空的情况分清楚,不然后面比较数值的时候就会操作空指针了。

节点为空的情况有:

  1. 1.左节点为空,右节点不为空,不对称,return false
  2. 左不为空,右为空,不对称 return false
  3. 左右都为空,对称,返回true

此时已经排除掉了节点为空的情况,那么剩下的就是左右节点不为空:左右都不为空,比较节点数值,不相同就return false

把以上情况都排除之后,剩下的就是 左右节点都不为空,且数值相同的情况

3.确定单层递归的逻辑

处理 左右节点都不为空,且数值相同的情况。

1.比较二叉树外侧是否对称:传入的是左节点的左孩子,右节点的右孩子。

2.比较内侧是否对称,传入左节点的右孩子,右节点的左孩子。

如果左右都对称就返回true ,有一侧不对称就返回false 。

 public boolean isSymmetric1(TreeNode root) {return compare(root.left, root.right);}private boolean compare(TreeNode left, TreeNode right) {1.确定递归函数的参数和返回值2.确定终止条件//处理节点为空的4种情况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;}3.确定单层递归的逻辑// 递归比较二叉树外侧boolean compareOutside = compare(left.left, right.right);// 递归比较内侧boolean compareInside = compare(left.right, right.left);return compareOutside && compareInside;//都为真则为真}

时间复杂度:O(n);(遍历二叉树)

空间复杂度:O(n);(递归树的高度h)

迭代法
public boolean isSymmetric3(TreeNode root) {Queue<TreeNode> deque = new LinkedList<>();deque.offer(root.left);deque.offer(root.right);while (!deque.isEmpty()) {TreeNode leftNode = deque.poll();TreeNode rightNode = deque.poll();if (leftNode == null && rightNode == null) {continue;}// 将其他三个判断条件合并if (leftNode == null || rightNode == null || leftNode.val != rightNode.val) {return false;}//左子树的左节点和右子树的右节点deque.offer(leftNode.left);deque.offer(rightNode.right);//左子树的右节点和右子树的左节点deque.offer(leftNode.right);deque.offer(rightNode.left);}return true;}

时间复杂度:O(n);(遍历二叉树)

空间复杂度:O(n);(使用队列)

以上是个人的思考反思与总结,若只想根据系列题刷,参考卡哥的网址代码随想录算法官网

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

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

相关文章

【研发日记】Matlab/Simulink技能解锁(一)——在Simulink编辑窗口Debug

文章目录 前言 时间阈值断点 信号阈值断点 周期步进 Signal Value Lable Data Inspector 分析和应用 总结 前言 近期在一些研发项目中使用Matlab/Simulink时&#xff0c;遇到了挺多费时费力的事情。所以利用晚上和周末时间&#xff0c;在这些方面深入研究了一下&#x…

GPT实战系列-LangChain的OutPutParser解析器

GPT实战系列-LangChain的OutPutParser解析器 LangChain GPT实战系列-LangChain如何构建基通义千问的多工具链 GPT实战系列-构建多参数的自定义LangChain工具 GPT实战系列-通过Basetool构建自定义LangChain工具方法 GPT实战系列-一种构建LangChain自定义Tool工具的简单方法…

leetcode 3.11

leetcode hot 100 二分查找1.寻找旋转排序数组中的最小值 矩阵1.搜索二维矩阵 II知识点&#xff1a;upper_bound, lower_bound知识点&#xff1a;二分查找 2.搜索二维矩阵 链表1.合并两个有序链表2.两数相加3. 删除链表的倒数第 N 个结点 二分查找 1.寻找旋转排序数组中的最小…

复习C语言基础中的基础:C语言发展、C89 C99有何区别、C语言特点

参考《C程序设计&#xff08;第五版&#xff09;》&#xff08;谭浩强&#xff09;一书&#xff1a; 1. 发展、C89 C99 2. 特点 记得时不时回顾一下背景特点&#xff0c;加深对C语言的理解。

git的实际运用

1. SSH配置和Github仓库克隆 注意博主在这里演示的SSH密钥生成方式&#xff0c;下面追加的五行不成功时可手动到.ssh下的config文件中添加即可 $ tail -5 config Host github.comHostName github.comPreferredAuthentications publickeyIdentityFile ~/.ssh/test 演示 2. 关联…

数据完整性

Oracle从入门到总裁:​​​​​​https://blog.csdn.net/weixin_67859959/article/details/135209645 数据完整性 数据完整性是关系数据库的一个重要特征&#xff0c;一般包含实体完整性、参照完整性和用户自定义完整性 3 种 实体完整性 实体完整性&#xff1a;规定表中的每…

QML| QML 组件

# | QML 组件 | 组件是可重用的、封装好的QML类型,并提供了定义好的接口。组件一般使用一个.qml文件定义。前面讲到的使用QML文档定义对象类型,其实就是创建了一个组件。这种使用独立QML文件创建组件的方法这里不再讨论。除了使用单独的QML文件,还可以使用Component类型在一…

如何规范员工上网行为(规范员工上网行为的有效方法)

有很多企业会有员工管理问题&#xff0c;比如以下几位老板的担忧&#xff1a; 可见&#xff0c;很多企业都想要对员工上网行为进行管理。 我想企业要对员工上网行为进行规范管理&#xff0c;肯定也是出于多方面的考量。这些考量可能涵盖了工作效率、信息安全、网络带宽利用以及…

2024 ICDE | 时间序列(Time Series)论文总结

第40届IEEE数据工程国际会议&#xff08;ICDE2024 &#xff09;于5月13日到17日在荷兰乌德勒支召开。 本届ICDE Research共接收论文375篇&#xff0c;Industry and Application接受论文21篇。本文总结了ICDE 2024上有关时间序列&#xff08;Time Series&#xff09;的工作&…

C语言-存储期2.0

静态存储期 在数据段中分配的变量&#xff0c;统统拥有静态存储期&#xff0c;因此也都被称为静态变量。这里静态的含义&#xff0c;指的是这些变量的不会因为程序的运行而发生临时性的分配和释放&#xff0c;它们的生命周期是恒定的&#xff0c;跟整个程序一致。 静态变量包含…

一文轻松学会远程服务器/docker内 vscode,调试(debug)无/多参数/bash以及多工作空间运行的python程序,欢迎大佬补充,一起学习

至于如何SSH、docker以及无需参数的程序调试这篇就先不说了。本篇文章主要记录一下多参数和多工作空间调试的问题。 一、launch.json文件 如上图所示&#xff0c;正常情况下当我们想要调试项目名字为0的目录下的train.py文件时&#xff0c;我们会按顺序点1&#xff0c;2&#…

防范服务器被攻击:查询IP地址的重要性与方法

在当今数字化时代&#xff0c;服务器扮演着重要的角色&#xff0c;为企业、组织和个人提供各种网络服务。然而&#xff0c;服务器也成为了网络攻击者的目标之一&#xff0c;可能面临各种安全威胁&#xff0c;例如DDoS攻击、恶意软件攻击、数据泄露等。为了有效地防范服务器被攻…

用 docker 创建 jmeter 容器, 实现性能测试,该如何下手?

用 docker 创建 jmeter 容器, 实现性能测试 我们都知道&#xff0c;jmeter可以做接口测试&#xff0c;也可以用于性能测试&#xff0c;现在企业中性能测试也大多使用jmeter。docker是最近这些年流行起来的容器部署工具&#xff0c;可以创建一个容器&#xff0c;然后把项目放到…

Navicat破解 Navicat下载安装 附教程 免费

百度网盘&#xff1a;https://pan.baidu.com/s/1wRRN_18_uXxPiIWCS4l43A 麻烦各位师傅帮忙填写一下问卷&#xff0c;提取码在问卷填写结束后显示~ 【https://www.wjx.cn/vm/mBBTTKm.aspx# 】 &#xff08;资料来源于网络&#xff0c;侵告删&#xff09;

【护网资料】 应急响应 安全设备 告警分析 蓝中必看 HVV 护网行动

百度网盘&#xff1a;https://pan.baidu.com/s/1wDVamgkyXwP-yiWN4jPEGQ 麻烦各位师傅帮忙填写一下问卷&#xff0c;提取码在问卷填写结束后显示~ 【https://www.wjx.cn/vm/mBBTTKm.aspx# 】 &#xff08;资料来源于网络&#xff0c;侵告删&#xff09;

【CTF笔记】 CTF web方向笔记分享 免费 附预览图

个人不怎么记东西&#xff0c;笔记不多&#xff0c;师傅们凑合看… 百度网盘&#xff1a;https://pan.baidu.com/s/1PspihUX28Y_AOQZPurHqKA 麻烦各位师傅帮忙填写一下问卷&#xff0c;提取码在问卷填写结束后显示~ 【https://www.wjx.cn/vm/mBBTTKm.aspx# 】 &#xff08;…

重学SpringBoot3-路径匹配机制

更多SpringBoot3内容请关注我的专栏&#xff1a;《SpringBoot3》 期待您的点赞&#x1f44d;收藏⭐评论✍ 重学SpringBoot3-路径匹配机制 AntPathMatcherPathPatternParser 和 PathPattern演示AntPathMatcher 示例PathPattern 示例性能和精确度的提升 选择使用哪一种 在 Spring…

迷宫问题三种种解法(A*算法+BFS+双向广搜)

题目描述 小明置身于一个迷宫&#xff0c;请你帮小明找出从起点到终点的最短路程。 小明只能向上下左右四个方向移动。 输入格式 输入包含多组测试数据。输入的第一行是一个整数T&#xff0c;表示有T组测试数据。 每组输入的第一行是两个整数N和M&#xff08;1<N,M<100&a…

【LeetCode】升级打怪之路 Day 24:回溯算法的解题框架

今日题目&#xff1a; 46. 全排列51. N 皇后78. 子集 目录 LC 46. 全排列LC 51. N 皇后LC 78. 子集 【classic】1&#xff09;思路一2&#xff09;思路二 今天学习了回溯算法的解题框架&#xff1a;回溯算法解题套路框架 | labuladong 回溯算法的整体框架都是&#xff1a; re…

力扣L11--- 344.反转字符串(JAVA版)-2024年3月15日

1.题目 2.知识点 交换两个变量值的代码 char temps[left];//temp为暂时的变量&#xff0c;left是左指针&#xff0c;将left暂时存储在temp里面s[left]s[right];//将右指针的值赋给左指针s[right]temp;//将temp的值给右指针left;//左指针向左移动right--;//右指针向右移动3.代码…