代码随想录-Day23

2024-05-30 01:20
文章标签 随想录 代码 day23

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

669. 修剪二叉搜索树

在这里插入图片描述

方法一:递归

class Solution {public TreeNode trimBST(TreeNode root, int low, int high) {if (root == null) {return null;}if (root.val < low) {return trimBST(root.right, low, high);} else if (root.val > high) {return trimBST(root.left, low, high);} else {root.left = trimBST(root.left, low, high);root.right = trimBST(root.right, low, high);return root;}}
}

这段代码定义了一个名为 Solution 的类,其中包含一个方法 trimBST,用于修剪(裁剪)给定二叉搜索树(BST)中的节点,使得所有节点的值在指定的区间 [low, high] 内。修剪操作应当保持二叉搜索树的性质。以下是代码逻辑的详细解析:

  1. 基本情况处理:首先检查根节点 root 是否为空。如果为空,直接返回 null,因为没有节点需要修剪。

  2. 节点值处理

    • 如果当前节点 root 的值 root.val 小于 low,说明当前节点及其左子树都不可能在保留范围内,因此直接递归地对右子树 root.right 调用 trimBST 方法,并返回结果作为新的根节点。这一步相当于“跳过”当前节点及其左子树。
    • 如果当前节点 root 的值 root.val 大于 high,说明当前节点及其右子树都不可能在保留范围内,因此直接递归地对左子树 root.left 调用 trimBST 方法,并返回结果作为新的根节点。这一步相当于“跳过”当前节点及其右子树。
    • 如果当前节点的值满足 low <= root.val <= high,说明当前节点值在指定区间内,需要保留。此时,递归地对左右子树进行修剪,并保持当前节点作为修剪后子树的根节点。
  3. 递归修剪子树:当当前节点值符合条件需要保留时,分别对左子树 root.left 和右子树 root.right 递归调用 trimBST 方法,以确保整个子树都被正确修剪。

  4. 返回处理后的节点:经过上述处理后,直接返回当前处理过的节点 root,作为修剪后子树的根。递归过程中,这个返回值会被上一层调用用来构建整个修剪后的BST。

通过这样的递归逻辑,trimBST 方法能够从根节点开始,逐步构建出只包含值在 [low, high] 范围内的二叉搜索树,并保持BST的性质。不在范围内的节点及其子树都会被“剪掉”,从而实现高效的修剪操作。

方法二:迭代

class Solution {public TreeNode trimBST(TreeNode root, int low, int high) {while (root != null && (root.val < low || root.val > high)) {if (root.val < low) {root = root.right;} else {root = root.left;}}if (root == null) {return null;}for (TreeNode node = root; node.left != null; ) {if (node.left.val < low) {node.left = node.left.right;} else {node = node.left;}}for (TreeNode node = root; node.right != null; ) {if (node.right.val > high) {node.right = node.right.left;} else {node = node.right;}}return root;}
}

这段代码提供了另一种实现方式,使用迭代方法来修剪二叉搜索树(BST),使其所有节点的值落在指定区间 [low, high] 内。相较于递归方法,迭代方法直接利用循环进行遍历和修剪。以下是代码逻辑的详细解析:

  1. 初始化:首先,代码通过一个 while 循环找到BST的第一个(最左边的)落在指定区间 [low, high] 内的节点作为新的根节点。如果根节点 root 的值小于 low,则向右移动(因为BST的性质保证了所有左子节点的值都小于根节点,所以要找大于等于 low 的节点,只能往右走);如果根节点的值大于 high,则向左移动。如果整个树的所有节点都不在区间内,最终 root 会变成 null,直接返回 null

  2. 修剪左子树:接下来,使用一个 for 循环来修剪根节点的左子树。循环的条件是当前节点的左子节点不为空。如果左子节点的值小于 low,说明整个左子树都不在指定区间内,直接将当前节点的左子节点更新为其左子节点的右子节点(跳过整个左子树的左部分);否则,将当前节点更新为其左子节点,继续检查更左的节点。

  3. 修剪右子树:随后,再使用一个类似的 for 循环来修剪根节点的右子树。循环的条件是当前节点的右子节点不为空。如果右子节点的值大于 high,说明整个右子树的右部分都不在指定区间内,直接将当前节点的右子节点更新为其右子节点的左子节点(跳过整个右子树的右部分);否则,将当前节点更新为其右子节点,继续检查更右的节点。

  4. 返回修剪后的根节点:经过上述两步修剪后,根节点及其左右子树都已经满足条件,直接返回 root 作为修剪后的二叉搜索树的根。

这种迭代方法同样保持了BST的性质,并且在处理每个节点时都是常量时间复杂度,总体时间复杂度为O(N),其中N为树中的节点数,空间复杂度为O(1),因为它只使用了固定数量的指针变量。

108. 将有序数组转换为二叉搜索树

给你一个整数数组 nums ,其中元素已经按 升序 排列,请你将其转换为一棵
平衡二叉搜索树。

方法一:中序遍历,总是选择中间位置左边的数字作为根节点

class Solution {public TreeNode sortedArrayToBST(int[] nums) {return helper(nums, 0, nums.length - 1);}public TreeNode helper(int[] nums, int left, int right) {if (left > right) {return null;}// 总是选择中间位置左边的数字作为根节点int mid = (left + right) / 2;TreeNode root = new TreeNode(nums[mid]);root.left = helper(nums, left, mid - 1);root.right = helper(nums, mid + 1, right);return root;}
}

这段代码定义了一个名为 Solution 的类,其中包含两个方法,用于将一个有序数组(升序)转换成一棵高度平衡的二叉搜索树(BST)。二叉搜索树的特性是左子树所有节点的值小于根节点的值,右子树所有节点的值大于根节点的值,且每个节点的左、右子树也分别是BST。下面是代码的详细解析:

  1. sortedArrayToBST(int[] nums) 方法:这是主要的接口方法,接收一个有序数组 nums 作为参数,然后调用辅助方法 helper 来完成转换工作,最终返回构建好的BST的根节点。

  2. helper(int[] nums, int left, int right) 方法:这是一个递归辅助方法,用于实际构建BST。

    • 输入参数nums 是原始有序数组,leftright 分别表示当前子数组的左右边界索引。
    • 终止条件:如果 left > right,表示当前子数组为空,没有节点可构建,因此返回 null
    • 选择根节点:为了构建高度平衡的BST,总是选择中间位置(或中间偏左,这里取中间下标 mid = (left + right) / 2)的元素作为根节点的值,这样做可以保证树尽量平衡。注意,当 leftright 为偶数时,mid 实际上取的是中间两个数中左边的那个。
    • 递归构建左右子树:以 mid 为界,分别对左半部分 [left, mid - 1] 和右半部分 [mid + 1, right] 递归调用 helper 方法,构建当前节点的左子树和右子树。
    • 返回根节点:构建好左右子树后,返回当前子树的根节点。

通过这样的递归划分,每个子数组都会被处理成一个高度平衡的BST子树,最终整个数组转换成了一棵高度平衡的二叉搜索树。这种方法充分利用了有序数组的特性,保证了构建的BST不仅是正确的,而且高度平衡,提高了树的查询效率。

方法二:中序遍历,总是选择中间位置右边的数字作为根节点

class Solution {public TreeNode sortedArrayToBST(int[] nums) {return helper(nums, 0, nums.length - 1);}public TreeNode helper(int[] nums, int left, int right) {if (left > right) {return null;}// 总是选择中间位置右边的数字作为根节点int mid = (left + right + 1) / 2;TreeNode root = new TreeNode(nums[mid]);root.left = helper(nums, left, mid - 1);root.right = helper(nums, mid + 1, right);return root;}
}

这段代码与之前提供的解决方案非常相似,都是将一个有序数组(升序)转换为一棵高度平衡的二叉搜索树(BST)。主要区别在于选取中间元素的方式:之前的解决方案选取中间位置左边的数字作为根节点,而这里的代码选择的是中间位置右边的数字。下面是代码解析:

  1. sortedArrayToBST(int[] nums) 方法:此方法作为接口,接收一个有序数组 nums,然后调用辅助方法 helper 来构建平衡BST,并返回根节点。

  2. helper(int[] nums, int left, int right) 方法:这是一个递归辅助方法,用于递归构建BST。

    • 输入参数nums 是原始有序数组,leftright 分别表示当前考虑构建子树的数组范围。
    • 终止条件:当 left > right 时,表示当前区间为空,无需构建节点,直接返回 null
    • 选择根节点:与之前版本不同,这里通过 (left + right + 1) / 2 计算中间索引,目的是选择区间的中间位置右边的数作为根节点。这确保了当数组长度为奇数时,中间值取右侧;偶数时,同样偏向取右侧的值作为根。
    • 递归构建子树:基于选定的根节点值,分别对左半区间 [left, mid - 1] 和右半区间 [mid + 1, right] 递归调用 helper 方法,构建当前节点的左子树和右子树。
    • 返回根节点:构建完左右子树后,返回当前子树的根节点。

通过这样的递归过程,整个数组被均衡地分割并构建为一棵高度平衡的BST,其中每个节点的值都来自数组中的一个位置,且树保持了BST的性质(左子树所有节点值小于根节点值,右子树所有节点值大于根节点值)。选择中间偏右的元素作为根节点是实现平衡的一种方式,虽不如选取正中间那样绝对平衡,但在大多数情况下能保持较好的平衡性。

方法三:中序遍历,选择任意一个中间位置数字作为根节点

class Solution {Random rand = new Random();public TreeNode sortedArrayToBST(int[] nums) {return helper(nums, 0, nums.length - 1);}public TreeNode helper(int[] nums, int left, int right) {if (left > right) {return null;}// 选择任意一个中间位置数字作为根节点int mid = (left + right + rand.nextInt(2)) / 2;TreeNode root = new TreeNode(nums[mid]);root.left = helper(nums, left, mid - 1);root.right = helper(nums, mid + 1, right);return root;}
}

这段代码依然致力于将一个有序数组(升序)转换为一棵高度平衡的二叉搜索树(BST),但与之前的实现有所不同,它在选择中间元素作为根节点时引入了随机性,以提高算法在特定输入下的性能表现。下面是代码的详细解析:

  1. 类成员变量:定义了一个 Random 类型的成员变量 rand,用于生成随机数。

  2. sortedArrayToBST(int[] nums) 方法:此方法与之前的实现相同,作为对外接口,接收有序数组并调用 helper 方法构建BST。

  3. helper(int[] nums, int left, int right) 方法:这个递归辅助方法负责实际的转换工作,其参数含义也保持一致。不同之处在于如何选择中间元素:

    • 随机选择中间位置:这里使用公式 (left + right + rand.nextInt(2)) / 2 来确定中间位置的索引。rand.nextInt(2) 会返回0或1,加上 leftright 后除以2,实质上会在左侧的中间位置和右侧的中间位置之间随机选择一个索引作为根节点。这种方法在理论上可以避免在特定输入数据下构造出极端不平衡的BST,比如当有序数组本身就是近乎有序的情况下,传统的总是选择中间位置作为根节点的方法可能导致构造出来的BST极度倾斜。通过随机化选择,可以使得构造的BST在统计学意义上更加平衡,提高树的操作效率,如查找、插入等。
  4. 构建左右子树:与之前的实现相同,根据选定的中间位置索引,递归地构建当前节点的左子树和右子树。

通过这种方式,尽管每次运行时可能生成不同的BST(因为根节点的选择是随机的),但整体上仍然能保证构建出的BST高度平衡,同时在一定程度上优化了在特定输入下的性能表现,特别是对于那些可能导致递归方法偏向构建非平衡树的特殊排序数组。

538. 把二叉搜索树转换为累加树

反序中序遍历

class Solution {int sum = 0;public TreeNode convertBST(TreeNode root) {if (root != null) {convertBST(root.right);sum += root.val;root.val = sum;convertBST(root.left);}return root;}
}

这段代码定义了一个名为 Solution 的类,其中包含一个方法 convertBST,该方法旨在将一个二叉搜索树(BST)转换成一个累加树。累加树是一种特殊的二叉树,其中每个节点的值等于原来的节点值加上所有大于它的节点值(在原BST中)。具体实现细节如下:

  • 类成员变量:

    • sum:这是一个整型成员变量,初始化为0,用于在遍历过程中累加节点值。
  • convertBST(TreeNode root) 方法

    • 输入参数:root 是二叉搜索树的根节点。
    • 返回值:返回转换后的二叉搜索树的根节点,现在它变成了一棵累加树。

方法内部逻辑遵循后序遍历的顺序(右子树 -> 根节点 -> 左子树),这是解决此类问题的关键,原因如下:

  1. 先遍历右子树:由于BST的性质(左子树的所有节点小于根节点,右子树的所有节点大于根节点),先访问右子树意味着我们从最大的节点开始累加,这符合累加树的要求。
  2. 累加当前节点值:在访问完当前节点的右子树后,将当前节点的值与已累加的值相加,然后更新当前节点的值。
  3. 遍历左子树:最后遍历左子树,这样在访问左子树的每个节点时,它们都会得到已经更新过的父节点及父节点右边所有节点的累加和。

通过这样的递归过程,整个BST被转换成了累加树,且每个节点的值都正确反映了累加的规则。最后,该方法返回转换后的树的根节点。

这篇关于代码随想录-Day23的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

python实现pdf转word和excel的示例代码

《python实现pdf转word和excel的示例代码》本文主要介绍了python实现pdf转word和excel的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价... 目录一、引言二、python编程1,PDF转Word2,PDF转Excel三、前端页面效果展示总结一

在MyBatis的XML映射文件中<trim>元素所有场景下的完整使用示例代码

《在MyBatis的XML映射文件中<trim>元素所有场景下的完整使用示例代码》在MyBatis的XML映射文件中,trim元素用于动态添加SQL语句的一部分,处理前缀、后缀及多余的逗号或连接符,示... 在MyBATis的XML映射文件中,<trim>元素用于动态地添加SQL语句的一部分,例如SET或W

使用C#代码计算数学表达式实例

《使用C#代码计算数学表达式实例》这段文字主要讲述了如何使用C#语言来计算数学表达式,该程序通过使用Dictionary保存变量,定义了运算符优先级,并实现了EvaluateExpression方法来... 目录C#代码计算数学表达式该方法很长,因此我将分段描述下面的代码片段显示了下一步以下代码显示该方法如

python多进程实现数据共享的示例代码

《python多进程实现数据共享的示例代码》本文介绍了Python中多进程实现数据共享的方法,包括使用multiprocessing模块和manager模块这两种方法,具有一定的参考价值,感兴趣的可以... 目录背景进程、进程创建进程间通信 进程间共享数据共享list实践背景 安卓ui自动化框架,使用的是

SpringBoot生成和操作PDF的代码详解

《SpringBoot生成和操作PDF的代码详解》本文主要介绍了在SpringBoot项目下,通过代码和操作步骤,详细的介绍了如何操作PDF,希望可以帮助到准备通过JAVA操作PDF的你,项目框架用的... 目录本文简介PDF文件简介代码实现PDF操作基于PDF模板生成,并下载完全基于代码生成,并保存合并P

SpringBoot基于MyBatis-Plus实现Lambda Query查询的示例代码

《SpringBoot基于MyBatis-Plus实现LambdaQuery查询的示例代码》MyBatis-Plus是MyBatis的增强工具,简化了数据库操作,并提高了开发效率,它提供了多种查询方... 目录引言基础环境配置依赖配置(Maven)application.yml 配置表结构设计demo_st

SpringCloud集成AlloyDB的示例代码

《SpringCloud集成AlloyDB的示例代码》AlloyDB是GoogleCloud提供的一种高度可扩展、强性能的关系型数据库服务,它兼容PostgreSQL,并提供了更快的查询性能... 目录1.AlloyDBjavascript是什么?AlloyDB 的工作原理2.搭建测试环境3.代码工程1.

Java调用Python代码的几种方法小结

《Java调用Python代码的几种方法小结》Python语言有丰富的系统管理、数据处理、统计类软件包,因此从java应用中调用Python代码的需求很常见、实用,本文介绍几种方法从java调用Pyt... 目录引言Java core使用ProcessBuilder使用Java脚本引擎总结引言python

Java中ArrayList的8种浅拷贝方式示例代码

《Java中ArrayList的8种浅拷贝方式示例代码》:本文主要介绍Java中ArrayList的8种浅拷贝方式的相关资料,讲解了Java中ArrayList的浅拷贝概念,并详细分享了八种实现浅... 目录引言什么是浅拷贝?ArrayList 浅拷贝的重要性方法一:使用构造函数方法二:使用 addAll(

JAVA利用顺序表实现“杨辉三角”的思路及代码示例

《JAVA利用顺序表实现“杨辉三角”的思路及代码示例》杨辉三角形是中国古代数学的杰出研究成果之一,是我国北宋数学家贾宪于1050年首先发现并使用的,:本文主要介绍JAVA利用顺序表实现杨辉三角的思... 目录一:“杨辉三角”题目链接二:题解代码:三:题解思路:总结一:“杨辉三角”题目链接题目链接:点击这里