力扣日记11.25-【二叉树篇】对称二叉树

力扣日记:【二叉树篇】对称二叉树

日期:2023.11.25
参考:代码随想录、力扣

101. 对称二叉树

题目描述

难度:简单

给你一个二叉树的根节点 root , 检查它是否轴对称。

示例 1:
在这里插入图片描述

输入:root = [1,2,2,3,4,4,3]
输出:true

示例 2:
在这里插入图片描述

输入:root = [1,2,2,null,3,null,3]
输出:false

提示:

  • 树中节点数目在范围 [1, 1000] 内
  • -100 <= Node.val <= 100

进阶:你可以运用递归和迭代两种方法解决这个问题吗?

题解

/*** Definition for a binary tree node.* struct TreeNode {*     int val;*     TreeNode *left;*     TreeNode *right;*     TreeNode() : val(0), left(nullptr), right(nullptr) {}*     TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}*     TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}* };*/
class Solution {
#define SOLUTION 1
public:
#if SOLUTION == 0bool isSymmetric(TreeNode* root) {/* // 错误的做法// 示例1:前:1234243, 中:3241423, 后:3424321,层:1223443// 示例2:中:12323,层:12233// 层序遍历不行,只能中序遍历// 使用栈,元素先进栈,遇到相同的则弹出// 还是不能用值来判断是否对称....如果树上节点的值都是相等的,那就无法判断了...stack<int> st_check;  // 用来判断// 中序遍历:左中右// 对于中序遍历,访问和处理并不是同步进行的。而是先访问到最底层的左节点,再开始处理(入栈判断)// 使用 cur 指针 先进行访问(遍历)// if (root == NULL) return true;stack<TreeNode*> st;    // 用来遍历TreeNode* cur = root;while (cur != NULL || !st.empty()) {if (cur != NULL) { // 指针访问节点,先遍历到最底层st.push(cur);   // 将cur入栈cur = cur->left;    // 左} else { // 处理cur = st.top();   // 中 (处理:放入result数组)st.pop();if (cur != root) {  // 根节点不入栈判断if (st_check.empty() || st_check.top() != cur->val) st_check.push(cur->val);    // 空或不相等则入栈else st_check.pop();    // 相同则弹出}cur = cur->right;   // 右 (如果右节点不为空,则在下次循环把右节点入栈;否则从栈中弹出顶部节点)}}return st_check.empty(); */}
#elif SOLUTION == 1     // 递归遍历/*思路:判断二叉树是否对称 -> 通过判断二叉树能否左右翻转分别比较外侧与内侧是否相等,即左节点的外侧(左孩子)需与对应右节点的外侧(右孩子)相等,内测同理采用的遍历方式为后序遍历 -> 后序:左右中 -> 在左右子树都判断好是否能左右翻转后,再将信息传递到父节点(中),中节点作为左孩子或右孩子又继续向上传递*/bool compare(TreeNode* left, TreeNode* right) { // 递归三要素1:参数与返回值:参数为当前层的左节点和对应右节点,返回值为两者的子树能否相互翻转(注意是子树,不包括自身,尽管只有当左和对应右节点相等才有继续比较的必要)// 递归三要素2:终止条件:// 1) 左右都为空:返回trueif (left == NULL && right == NULL)  return true;// 2) 左为空右不为空 或 左不为空右为空:返回falseelse if ((left != NULL && right == NULL) || (left == NULL && right != NULL))    return false;// 3) 左右不为空且左右值不相等else if (left->val != right->val)   return false;// 递归三要素3:处理逻辑:如果左右值相等,则递归向下判断else {                                                  // 由此可见:左右子树均为后序遍历// 外侧:bool outside = compare(left->left, right->right);   // 左子树:左、 右子树:右// 内侧:bool inside = compare(left->right, right->left);    // 左子树:右、 右子树:左// 将外侧内侧的比较结果向上传递(给中节点)bool isSame = outside && inside;                    // 左子树:中、 右子树:中 (逻辑处理)return isSame; // 当外侧和内侧的子节点都分别相等,则当前left和right的子树是可以翻转的}}bool isSymmetric(TreeNode* root) {return compare(root->left, root->right);}#elif SOLUTION == 2 // 迭代法(队列)bool isSymmetric(TreeNode* root) {// 思路:// 遍历:将左侧节点和对应右侧节点成对放入队列;// 处理:再在弹出时成对弹出比较是否相等,相等则继续遍历子节点,否则终止queue<TreeNode*> q;if (root != nullptr) {  // 先把根节点的左右节点放入队列q.push(root->left);q.push(root->right);}while (!q.empty()) {// 成对弹出TreeNode* leftSide = q.front();     q.pop();TreeNode* rightSide = q.front();    q.pop();// 比较if (!leftSide && !rightSide) continue; // 左右都为空(没有子节点,则继续弹出)// 左右有一个为空 或 左右都不为空但不相等,则肯定不对称,返回falseelse if (!leftSide || !rightSide || (leftSide->val != rightSide->val)) return false;// 如果相等,则继续遍历,将子节点入队列q.push(leftSide->left);q.push(rightSide->right); // 注意要成对:左的左 与 右的右q.push(leftSide->right);q.push(rightSide->left);    // 左的右 与 右的左}return true;}
#elif SOLUTION == 3 // 迭代法(栈)bool isSymmetric(TreeNode* root) {// 思路与队列类似,也是成对入栈、成对弹出、成对比较(队列更好理解……)stack<TreeNode*> st;if (root != nullptr) {  // 先把根节点的左右节点放入队列st.push(root->left);st.push(root->right);}while (!st.empty()) {// 成对弹出TreeNode* rightSide = st.top();  st.pop();TreeNode* leftSide = st.top();   st.pop();     // 比较if (!leftSide && !rightSide) continue; // 左右都为空(没有子节点,则继续弹出)// 左右有一个为空 或 左右都不为空但不相等,则肯定不对称,返回falseelse if (!leftSide || !rightSide || (leftSide->val != rightSide->val)) return false;// 如果相等,则继续遍历,将子节点入队列st.push(leftSide->left);st.push(rightSide->right); // 注意要成对:左的左 与 右的右st.push(leftSide->right);st.push(rightSide->left);    // 左的右 与 右的左}return true;}
#endif
};

复杂度

时间复杂度:
空间复杂度:

思路总结

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

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

相关文章

操作系统题目分类总结 | 进程管理 内存管理 文件系统 设备管理

系列文章如下 学习过程中一定要有系统观念&#xff08;知识框架&#xff0c;每一章开头都会有一个思维导图&#xff09;&#xff0c;知道目前自己在学习的是哪一板块的内容&#xff0c;和前面有什么样的联系 操作系统的很多知识点前后都是联系非常紧密的&#xff0c;去一点一…

PCL 计算一条直线与一条线段的距离

文章目录 一、简介二、实现代码三、实现效果参考资料一、简介 这里的线段我们仍然使用一种参数化的表示,即 Q [ 0 ] + s [ 0 ] ∗ ( Q [ 1 ] − Q [ 0

Blender 连续 5 天遭受大规模 DDoS 攻击

Blender 发布公告指出&#xff0c;在2023年11月18日至23日期间&#xff0c;blender.org 网站遭受了持续的分布式拒绝服务&#xff08;DDoS&#xff09;攻击&#xff0c;攻击者通过不断发送请求导致服务器超载&#xff0c;使网站运营严重中断。此次攻击涉及数百个 IP 地址的僵尸…

ATK-ESP8266 WIFI模块串口通信通用实现方案

ATK-ESP8266 WIFI模块是一种常用的无线模块&#xff0c;它可以通过串口与外部设备进行通信&#xff0c;实现数据的收发和控制。本文将介绍一种通用的实现方案&#xff0c;帮助您在项目中使用ATK-ESP8266 WIFI模块进行串口通信。 【方案概述】 这个通用实现方案涵盖了ATK-ESP82…

springboot(ssm付费自习室管理系统 自习室预约平台Java(codeLW)

springboot(ssm付费自习室管理系统 自习室预约平台Java(code&LW) 开发语言&#xff1a;Java 框架&#xff1a;ssm/springboot vue JDK版本&#xff1a;JDK1.8&#xff08;或11&#xff09; 服务器&#xff1a;tomcat 数据库&#xff1a;mysql 5.7&#xff08;或8.0&am…

算法-技巧-中等-颜色分类

记录一下算法题的学习12 颜色分类 题目&#xff1a;给定一个包含红色、白色和蓝色、共 n 个元素的数组 nums &#xff0c;原地对它们进行排序&#xff0c;使得相同颜色的元素相邻&#xff0c;并按照红色、白色、蓝色顺序排列。 我们使用整数 0、 1 和 2 分别表示红色、白色和蓝…

Android 相机库CameraView源码解析 (二) : 拍照

1. 前言 这段时间&#xff0c;在使用 natario1/CameraView 来实现带滤镜的预览、拍照、录像功能。 由于CameraView封装的比较到位&#xff0c;在项目前期&#xff0c;的确为我们节省了不少时间。 但随着项目持续深入&#xff0c;对于CameraView的使用进入深水区&#xff0c;逐…

了解JSX

在React中使用JSX <!DOCTYPE html> <html> <head> <meta charset"UTF-8" /> <title>JSX</title> </head> <body> JSX <!-- JSX 概念&#xff1a;JSX是JavaScript和XML的缩写&#xff0c;表示在js代码中编写…

初次使用vs code时go模块安装失败的解决办法

问题分析 go语言在vs code中下载模块时&#xff0c;会使用自己的代理&#xff0c;这个代理在大陆无法访问&#xff0c;需要将代理修改为国内的&#xff0c;模块就可以正常下载了&#xff0c;阿里的代理我试过了&#xff0c;有问题。 解决步骤 确保已经配置了go的环境变量&am…

开源与闭源:大模型未来的发展之争

在当今数字化时代&#xff0c;开源与闭源软件一直是技术界争论的热点话题。随着人工智能技术的快速发展&#xff0c;特别是大模型&#xff08;如GPT-4等&#xff09;的广泛应用&#xff0c;这个辩论在大模型技术的背景下变得更加引人注目。本文将探讨开源与闭源的优劣势比较&am…

WordPress无需插件禁用WP生成1536×1536和2048×2048尺寸图片

我们在使用WordPress上传图片媒体文件的时候&#xff0c;是不是看到媒体库中有15361536和20482048的图片文件&#xff0c;当然这么大的文件会占用我们的服务器空间&#xff0c;如何禁止掉呢&#xff1f; function remove_default_image_sizes( $sizes) {unset( $sizes[1536x15…

spring-webmvc练习-日程管理-访问后端展示列表数据

1、util/request.js import axios from "axios";let request axios.create({baseURL: "http://localhost:8080",timeout: 50000 });export default request 2、api/schedule.js import request from "../util/request.js";export let getSchedu…

[架构之路-253]:目标系统 - 设计方法 - 软件工程 - 软件设计 - 结构化设计的主要评估指标:高内聚(模块内部)、低耦合(模块之间)的含义

目录 前言&#xff1a; 一、软件工程中的软件设计种类&#xff1a;根据宏观到微观分 &#xff08;1&#xff09;软件架构设计&#xff08;层次划分、模块划分、职责分工&#xff09;&#xff1a; &#xff08;2&#xff09;软件高层设计、概要设计&#xff08;功能模块的接…

【c++随笔14】虚函数表

【c随笔14】虚函数表 一、虚函数表&#xff08;Virtual Function Table&#xff09;1、定义2、查看虚函数表2.1、 问题&#xff1a;三种类型&#xff0c;包含一个int类型的class、一个int类型的变量、int类型的指针&#xff1a;这三个大小分别是多少呢&#xff1f;2.2、怎么发现…

Vue项目创建

1.首先按WinR&#xff0c;输入cmd&#xff0c;开启命令行 2.跳转到项目目标位置&#xff0c;比如我的项目创建位置是E:\Vue-test&#xff0c;因而输入&#xff1a; e: cd E:\Vue-test 3.输入npm init vuelatest&#xff08;初始化Vue&#xff09;&#xff0c;如果是第一次用…

IT问题解答类型网站源码

问答网是一款为IT工程师提供的问答平台&#xff0c;旨在帮助用户在线获取专业知识和相关问题的答案。在问答网&#xff0c;用户可以轻松找到其他人的问答问题&#xff0c;并在这里寻求解答。如果您有任何想要解决的问题&#xff0c;都可以在此发布问题并得到其他同行的解答。 …

kafka kraft 集群搭建保姆级教学 包含几个踩坑点

一.为啥弃用zookeeper kafka 弃用 ZooKeeper 而采用 KRaft 的主要原因是为了改进 Kafka 集群的可靠性和可管理性。 在传统的 Kafka 架构中&#xff0c;ZooKeeper 用于存储和管理集群的元数据、配置信息和状态。然而&#xff0c;使用 ZooKeeper 作为协调服务存在一些限制和挑战…

CSS之弹性盒子Flexible Box

我想大家在做布局的时候&#xff0c;没接触flex布局之前&#xff0c;大家都是用浮动来布局的&#xff0c;但现在我们接触了flex布局之后&#xff0c;我只能说&#xff1a;“真香”。让我为大家介绍一下弹性盒子模型吧&#xff01; Flexible Box 弹性盒子 在我们使用弹性盒子时&…

bootstrap 5 登录、注册页面

bootstrap 5 登录、注册页面 <!DOCTYPE html> <html lang"en"> <head> <meta charset"UTF-8"> <meta name"viewport" content"widthdevice-width, initial-scale1.0"> <title>Login Page…

【算法】链表-20231127

这里写目录标题 一、面试题 02.02. 返回倒数第 k 个节点二、82. 删除排序链表中的重复元素 II三、141. 环形链表 一、面试题 02.02. 返回倒数第 k 个节点 提示 简单 130 相关企业 实现一种算法&#xff0c;找出单向链表中倒数第 k 个节点。返回该节点的值。 注意&#xff1a;本…