代码随想录 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调用DeepSeek API的最佳实践及详细代码示例

《Java调用DeepSeekAPI的最佳实践及详细代码示例》:本文主要介绍如何使用Java调用DeepSeekAPI,包括获取API密钥、添加HTTP客户端依赖、创建HTTP请求、处理响应、... 目录1. 获取API密钥2. 添加HTTP客户端依赖3. 创建HTTP请求4. 处理响应5. 错误处理6.

使用 sql-research-assistant进行 SQL 数据库研究的实战指南(代码实现演示)

《使用sql-research-assistant进行SQL数据库研究的实战指南(代码实现演示)》本文介绍了sql-research-assistant工具,该工具基于LangChain框架,集... 目录技术背景介绍核心原理解析代码实现演示安装和配置项目集成LangSmith 配置(可选)启动服务应用场景

Python中顺序结构和循环结构示例代码

《Python中顺序结构和循环结构示例代码》:本文主要介绍Python中的条件语句和循环语句,条件语句用于根据条件执行不同的代码块,循环语句用于重复执行一段代码,文章还详细说明了range函数的使... 目录一、条件语句(1)条件语句的定义(2)条件语句的语法(a)单分支 if(b)双分支 if-else(

MySQL数据库函数之JSON_EXTRACT示例代码

《MySQL数据库函数之JSON_EXTRACT示例代码》:本文主要介绍MySQL数据库函数之JSON_EXTRACT的相关资料,JSON_EXTRACT()函数用于从JSON文档中提取值,支持对... 目录前言基本语法路径表达式示例示例 1: 提取简单值示例 2: 提取嵌套值示例 3: 提取数组中的值注意

CSS3中使用flex和grid实现等高元素布局的示例代码

《CSS3中使用flex和grid实现等高元素布局的示例代码》:本文主要介绍了使用CSS3中的Flexbox和Grid布局实现等高元素布局的方法,通过简单的两列实现、每行放置3列以及全部代码的展示,展示了这两种布局方式的实现细节和效果,详细内容请阅读本文,希望能对你有所帮助... 过往的实现方法是使用浮动加

JAVA调用Deepseek的api完成基本对话简单代码示例

《JAVA调用Deepseek的api完成基本对话简单代码示例》:本文主要介绍JAVA调用Deepseek的api完成基本对话的相关资料,文中详细讲解了如何获取DeepSeekAPI密钥、添加H... 获取API密钥首先,从DeepSeek平台获取API密钥,用于身份验证。添加HTTP客户端依赖使用Jav

Java实现状态模式的示例代码

《Java实现状态模式的示例代码》状态模式是一种行为型设计模式,允许对象根据其内部状态改变行为,本文主要介绍了Java实现状态模式的示例代码,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来... 目录一、简介1、定义2、状态模式的结构二、Java实现案例1、电灯开关状态案例2、番茄工作法状态案例

nginx-rtmp-module模块实现视频点播的示例代码

《nginx-rtmp-module模块实现视频点播的示例代码》本文主要介绍了nginx-rtmp-module模块实现视频点播,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习... 目录预置条件Nginx点播基本配置点播远程文件指定多个播放位置参考预置条件配置点播服务器 192.

CSS自定义浏览器滚动条样式完整代码

《CSS自定义浏览器滚动条样式完整代码》:本文主要介绍了如何使用CSS自定义浏览器滚动条的样式,包括隐藏滚动条的角落、设置滚动条的基本样式、轨道样式和滑块样式,并提供了完整的CSS代码示例,通过这些技巧,你可以为你的网站添加个性化的滚动条样式,从而提升用户体验,详细内容请阅读本文,希望能对你有所帮助...

深入解析Spring TransactionTemplate 高级用法(示例代码)

《深入解析SpringTransactionTemplate高级用法(示例代码)》TransactionTemplate是Spring框架中一个强大的工具,它允许开发者以编程方式控制事务,通过... 目录1. TransactionTemplate 的核心概念2. 核心接口和类3. TransactionT