代码随想录 day21 第六章 二叉树part07

2024-03-14 09:28

本文主要是介绍代码随想录 day21 第六章 二叉树part07,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

今日内容

●  530.二叉搜索树的最小绝对差

●  501.二叉搜索树中的众数

●  236. 二叉树的最近公共祖先

详细布置

1. 二叉搜索树的最小绝对差

关联 leetcode 530.二叉搜索树的最小绝对差

  • 思路
    • 直白想法
      • 中序遍历可以得到一个有序节点值序列
        1. 先中序排列二叉树
        2. 暴力找最小差
    • 双指针
      • 中序遍历二叉树+双指针
        • 左【pre】+中【cur】
        • 中【pre】+右【cur】
  • 题解
    • 递归

      func getMinimumDifference(root *TreeNode) int {res := math.MaxIntvar pre *TreeNode = nilvar traversal func(cur *TreeNode)traversal = func(cur *TreeNode) {if cur == nil {return}//左traversal(cur.Left)//中if pre != nil {res = min(res, cur.Val-pre.Val)}pre = cur//中是相对于右而言的pre//右traversal(cur.Right)}traversal(root)return res}
      
    • 迭代

      func getMinimumDifference(root *TreeNode) int {stack := make([]*TreeNode, 0) //存储节点的栈var cur *TreeNode = root      //当前节点指针var pre *TreeNode = nil       //前一个节点的指针res := math.MaxIntfor cur != nil || len(stack) > 0 {if cur != nil { // 指针来访问节点,访问到最底层stack = append(stack, cur) // 将访问的节点放进栈cur = cur.Left             //左} else {cur = stack[len(stack)-1] //取出栈底元素,左左元素stack = stack[:len(stack)-1]if pre != nil {res = min(res, cur.Val-pre.Val)}pre = curcur = cur.Right}}return res
      }
      

2. 二叉搜索树中的众数

关联 leetcode 501.二叉搜索树中的众数

和 530差不多双指针思路,不过 这里涉及到一个很巧妙的代码技巧。

可以先自己做做看,然后看我的视频讲解。

https://programmercarl.com/0501.二叉搜索树中的众数.html

视频讲解:不仅双指针,还有代码技巧可以惊艳到你! | LeetCode:501.二叉搜索树中的众数_哔哩哔哩_bilibili

  • 题解
    • 自己解法

      • 使用一个map来存储 元素出现次数
      /*** Definition for a binary tree node.* type TreeNode struct {*     Val int*     Left *TreeNode*     Right *TreeNode* }*/
      func findMode(root *TreeNode) []int {rets := make([]int, 0)countMap := make(map[int]int)var traversal func(root *TreeNode)traversal = func(root *TreeNode) {if root == nil {return}countMap[root.Val]++if root.Left != nil {traversal(root.Left)}if root.Right != nil {traversal(root.Right)}}traversal(root)maxCount := 0for k, v := range countMap {if v > maxCount {maxCount = vrets = make([]int, 0)rets = append(rets, k)} else if v == maxCount {rets = append(rets, k)}}return rets
      }
      
    • 二叉搜索树一定是中序遍历

      • 遍历得到的二叉树节点一定是有序的
      • 讲解双指针法
      • 和自己的思路不差, 省去了map的使用, 提升效率
      func findMode(root *TreeNode) []int {ret := make([]int, 0)var pre *TreeNode = nilcount := 0maxCount := 0var traversal func(cur *TreeNode)traversal = func(cur *TreeNode) {if cur == nil {return}traversal(cur.Left) //Left/* Middle */if pre == nil {count = 1 //初始化遍历, 首次走到叶子节点} else if pre.Val == cur.Val {count++ //找到重复元素, 当前元素出现次数递加} else {count = 1 //当前元素是新出现的元素, 重置出现次数}pre = cur //将前一个元素指针指向当前元素if count == maxCount {ret = append(ret, cur.Val)}if count > maxCount {maxCount = countret = []int{cur.Val}}traversal(cur.Right) //Right}traversal(root)return ret
      }
      

3. 二叉树的最近公共祖先

关联 leetcode 236. 二叉树的最近公共祖先

  • 思路

    • 题目强调
      • 输入节点的值互不相同
      • p、q节点一定存在于所给二叉树中
        • 一定能找到祖先节点,最终结果一定不是nil
    • 公共祖先
      • 祖先的左、右节点或者它自己分别包含一个指定节点即可
      • 从底向上回溯查找即可
        • 选用后序遍历
          • 先找左、右孩子,再处理父亲节点本身
            • 如果左右子树恰好包含要找的元素,这时的父亲节点就是一个祖先
  • 题解

    func lowestCommonAncestor(root, p, q *TreeNode) *TreeNode {if root == nil {return nil}if root == q || root == p {return root}//左子树left := lowestCommonAncestor(root.Left, p, q)//右子树right := lowestCommonAncestor(root.Right, p, q)//中if left != nil || right != nil {//当前节点的左子树,右子树恰好包含元素,该节点就是最近公共祖先return root}//走到下面就是至少左右有一个是nilif right != nil {//说明目标节点是通过right返回的return right}if left != nil  {//说明目标节点是通过left返回的return left}  return nil}
    

9. 题外话

  1. 求最小公共祖先,需要从底向上遍历,那么二叉树,只能通过后序遍历(即:回溯)实现从底向上的遍历方式。
  2. 在回溯的过程中,必然要遍历整棵二叉树,即使已经找到结果了,依然要把其他节点遍历完,因为要使用递归函数的返回值(也就是代码中的left和right)做逻辑判断。
  3. 要理解如果返回值left为空,right不为空为什么要返回right,为什么可以用返回right传给上一层结果。

这篇关于代码随想录 day21 第六章 二叉树part07的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/807967

相关文章

Java中调用数据库存储过程的示例代码

《Java中调用数据库存储过程的示例代码》本文介绍Java通过JDBC调用数据库存储过程的方法,涵盖参数类型、执行步骤及数据库差异,需注意异常处理与资源管理,以优化性能并实现复杂业务逻辑,感兴趣的朋友... 目录一、存储过程概述二、Java调用存储过程的基本javascript步骤三、Java调用存储过程示

Visual Studio 2022 编译C++20代码的图文步骤

《VisualStudio2022编译C++20代码的图文步骤》在VisualStudio中启用C++20import功能,需设置语言标准为ISOC++20,开启扫描源查找模块依赖及实验性标... 默认创建Visual Studio桌面控制台项目代码包含C++20的import方法。右键项目的属性:

MySQL数据库的内嵌函数和联合查询实例代码

《MySQL数据库的内嵌函数和联合查询实例代码》联合查询是一种将多个查询结果组合在一起的方法,通常使用UNION、UNIONALL、INTERSECT和EXCEPT关键字,下面:本文主要介绍MyS... 目录一.数据库的内嵌函数1.1聚合函数COUNT([DISTINCT] expr)SUM([DISTIN

Java实现自定义table宽高的示例代码

《Java实现自定义table宽高的示例代码》在桌面应用、管理系统乃至报表工具中,表格(JTable)作为最常用的数据展示组件,不仅承载对数据的增删改查,还需要配合布局与视觉需求,而JavaSwing... 目录一、项目背景详细介绍二、项目需求详细介绍三、相关技术详细介绍四、实现思路详细介绍五、完整实现代码

Go语言代码格式化的技巧分享

《Go语言代码格式化的技巧分享》在Go语言的开发过程中,代码格式化是一个看似细微却至关重要的环节,良好的代码格式化不仅能提升代码的可读性,还能促进团队协作,减少因代码风格差异引发的问题,Go在代码格式... 目录一、Go 语言代码格式化的重要性二、Go 语言代码格式化工具:gofmt 与 go fmt(一)

HTML5实现的移动端购物车自动结算功能示例代码

《HTML5实现的移动端购物车自动结算功能示例代码》本文介绍HTML5实现移动端购物车自动结算,通过WebStorage、事件监听、DOM操作等技术,确保实时更新与数据同步,优化性能及无障碍性,提升用... 目录1. 移动端购物车自动结算概述2. 数据存储与状态保存机制2.1 浏览器端的数据存储方式2.1.

基于 HTML5 Canvas 实现图片旋转与下载功能(完整代码展示)

《基于HTML5Canvas实现图片旋转与下载功能(完整代码展示)》本文将深入剖析一段基于HTML5Canvas的代码,该代码实现了图片的旋转(90度和180度)以及旋转后图片的下载... 目录一、引言二、html 结构分析三、css 样式分析四、JavaScript 功能实现一、引言在 Web 开发中,

Python如何去除图片干扰代码示例

《Python如何去除图片干扰代码示例》图片降噪是一个广泛应用于图像处理的技术,可以提高图像质量和相关应用的效果,:本文主要介绍Python如何去除图片干扰的相关资料,文中通过代码介绍的非常详细,... 目录一、噪声去除1. 高斯噪声(像素值正态分布扰动)2. 椒盐噪声(随机黑白像素点)3. 复杂噪声(如伪

Java Spring ApplicationEvent 代码示例解析

《JavaSpringApplicationEvent代码示例解析》本文解析了Spring事件机制,涵盖核心概念(发布-订阅/观察者模式)、代码实现(事件定义、发布、监听)及高级应用(异步处理、... 目录一、Spring 事件机制核心概念1. 事件驱动架构模型2. 核心组件二、代码示例解析1. 事件定义

Python实例题之pygame开发打飞机游戏实例代码

《Python实例题之pygame开发打飞机游戏实例代码》对于python的学习者,能够写出一个飞机大战的程序代码,是不是感觉到非常的开心,:本文主要介绍Python实例题之pygame开发打飞机... 目录题目pygame-aircraft-game使用 Pygame 开发的打飞机游戏脚本代码解释初始化部