热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Java详解剑指offer面试题27对称的二叉树

Java详解剑指offer面试题27–对称的二叉树请实现一个函数,用来判断一颗二叉树是不是对称的。注意,如果一个二叉树同此二叉树的镜像是同样的ÿ
Java详解剑指offer面试题27–对称的二叉树

请实现一个函数,用来判断一颗二叉树是不是对称的。注意,如果一个二叉树同此二叉树的镜像是同样的,定义其为对称的。

对称的二叉树,结点数必然是奇数,特别地定义空树也是对称的。当对称树的结点值不是完全相同时比较好处理,但是当结点值全部一样时候可能会有些麻烦。我们要实现一个通用的算法,使得对于这种特殊情况也能正确处理。

要保证树是对称的,左子树最左边的结点要和右子树最右边的结点值相同…左子树的右子结点要和右子树的左子结点值相同,即

root1.left == root2.right;
root1.right == root2.left;

这是关键,理解了上面说的,可以很轻松地写出下面的代码。

递归版本

先来看递归版本,递归的实现一般比较简单、清晰。

package Chap4;import java.util.LinkedList;
import java.util.Queue;public class SymmetricalTree {public class TreeNode {int val = 0;TreeNode left = null;TreeNode right = null;public TreeNode(int val) {this.val = val;}}/*** 递归实现*/public boolean isSymmetricalRecur(TreeNode pRoot) {if (pRoot == null) return true;return isSymmetrical(pRoot.left, pRoot.right);}private boolean isSymmetrical(TreeNode root1, TreeNode root2) {if (root1 == null && root2 == null) return true;if (root1 == null || root2 == null) return false;if (root1.val != root2.val) return false;return isSymmetrical(root1.left, root2.right)&& isSymmetrical(root1.right, root2.left);}
}

如果遇到两棵子树都为空流返回true,这句代码隐含了**“空树也是对称的”**这样的信息。否则,只有一子树为空另一不为空,显然不是对称的;如果两个子树都不为空,比较它俩根结点的值,不相同肯定不是对称的。之后递归地对树的子树进行上述判断,直到检查到叶子结点,如果都满足就返回true。

非递归版本

思路和上面一样,非递归实现需要用到两个队列。队列A专门存入左子树,队列B专门存入右子树。

入列时,将左子树的左子结点和右子树的右子结点分别存入队列A和队列B,紧接着还要将左子树的右子结点和右子树的左子结点分别存入队列A和队列B。

出列时,两个队列同时出列一个元素,根据存入的顺序,这两个出列元素就是左子树的左子结点和右子树的右子结点,或者左子树的右子结点和右子树的左子结点。之后对这两个元素进行比较,比较操作和上面递归版本的一样。

package Chap4;import java.util.LinkedList;
import java.util.Queue;public class SymmetricalTree {/*** 非递归&#xff0c;队列实现(栈也可以实现)*/public boolean isSymmetrical(TreeNode pRoot) {if (pRoot &#61;&#61; null) return true;Queue<TreeNode> queueA &#61; new LinkedList<>();Queue<TreeNode> queueB &#61; new LinkedList<>();queueA.offer(pRoot.left);queueB.offer(pRoot.right);TreeNode left &#61; null;TreeNode right &#61; null;while (!queueA.isEmpty() && !queueB.isEmpty()) {left &#61; queueA.poll();right &#61; queueB.poll();// 两个都空跳过if (left &#61;&#61; null && right &#61;&#61; null) continue;// 只有一个空&#xff0c;不对称if (left &#61;&#61; null || right &#61;&#61; null) return false;// 两个都不空&#xff0c;比较值if (left.val !&#61; right.val) return false;// 两两对称的加入// 左孩子的左孩子&#xff0c;右孩子的右孩子queueA.offer(left.left);queueB.offer(right.right);// 左孩子的右孩子&#xff0c;右孩子的左孩子queueA.offer(left.right);queueB.offer(right.left);}return true;}
}


本文参考文献&#xff1a;
[1]github.com/haiyusun/data-structures


推荐阅读
  • 欢乐的票圈重构之旅——RecyclerView的头尾布局增加
    项目重构的Git地址:https:github.comrazerdpFriendCircletreemain-dev项目同步更新的文集:http:www.jianshu.comno ... [详细]
  • 云原生边缘计算之KubeEdge简介及功能特点
    本文介绍了云原生边缘计算中的KubeEdge系统,该系统是一个开源系统,用于将容器化应用程序编排功能扩展到Edge的主机。它基于Kubernetes构建,并为网络应用程序提供基础架构支持。同时,KubeEdge具有离线模式、基于Kubernetes的节点、群集、应用程序和设备管理、资源优化等特点。此外,KubeEdge还支持跨平台工作,在私有、公共和混合云中都可以运行。同时,KubeEdge还提供数据管理和数据分析管道引擎的支持。最后,本文还介绍了KubeEdge系统生成证书的方法。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 安卓select模态框样式改变_微软Office风格的多端(Web、安卓、iOS)组件库——Fabric UI...
    介绍FabricUI是微软开源的一套Office风格的多端组件库,共有三套针对性的组件,分别适用于web、android以及iOS,Fab ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • JVM 学习总结(三)——对象存活判定算法的两种实现
    本文介绍了垃圾收集器在回收堆内存前确定对象存活的两种算法:引用计数算法和可达性分析算法。引用计数算法通过计数器判定对象是否存活,虽然简单高效,但无法解决循环引用的问题;可达性分析算法通过判断对象是否可达来确定存活对象,是主流的Java虚拟机内存管理算法。 ... [详细]
  • sklearn数据集库中的常用数据集类型介绍
    本文介绍了sklearn数据集库中常用的数据集类型,包括玩具数据集和样本生成器。其中详细介绍了波士顿房价数据集,包含了波士顿506处房屋的13种不同特征以及房屋价格,适用于回归任务。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • importjava.util.ArrayList;publicclassPageIndex{privateintpageSize;每页要显示的行privateintpageNum ... [详细]
  • 自动轮播,反转播放的ViewPagerAdapter的使用方法和效果展示
    本文介绍了如何使用自动轮播、反转播放的ViewPagerAdapter,并展示了其效果。该ViewPagerAdapter支持无限循环、触摸暂停、切换缩放等功能。同时提供了使用GIF.gif的示例和github地址。通过LoopFragmentPagerAdapter类的getActualCount、getActualItem和getActualPagerTitle方法可以实现自定义的循环效果和标题展示。 ... [详细]
  • 个人学习使用:谨慎参考1Client类importcom.thoughtworks.gauge.Step;importcom.thoughtworks.gauge.T ... [详细]
  • 标题: ... [详细]
  • Android工程师面试准备及设计模式使用场景
    本文介绍了Android工程师面试准备的经验,包括面试流程和重点准备内容。同时,还介绍了建造者模式的使用场景,以及在Android开发中的具体应用。 ... [详细]
  • 本文介绍了在Python3中如何使用选择文件对话框的格式打开和保存图片的方法。通过使用tkinter库中的filedialog模块的asksaveasfilename和askopenfilename函数,可以方便地选择要打开或保存的图片文件,并进行相关操作。具体的代码示例和操作步骤也被提供。 ... [详细]
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
author-avatar
美食和旅丶行_379
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有