首页 星云 工具 资源 星选 资讯 热门工具
:

PDF转图片 完全免费 小红书视频下载 无水印 抖音视频下载 无水印 数字星空

代码随想录Day21

编程知识
2024年08月20日 20:48

669. 修剪二叉搜索树

给你二叉搜索树的根节点 root ,同时给定最小边界low 和最大边界 high。通过修剪二叉搜索树,使得所有节点的值在[low, high]中。修剪树 不应该 改变保留在树中的元素的相对结构 (即,如果没有被移除,原有的父代子代关系都应当保留)。 可以证明,存在 唯一的答案 。

所以结果应当返回修剪好的二叉搜索树的新的根节点。注意,根节点可能会根据给定的边界发生改变。

示例 1:

输入:root = [1,0,2], low = 1, high = 2
输出:[1,null,2]

示例 2:

输入:root = [3,0,4,null,2,null,null,1], low = 1, high = 3
输出:[3,2,null,1]

提示:

树中节点数在范围 [1, 104] 内
0 <= Node.val <= 104
树中每个节点的值都是 唯一 的
题目数据保证输入是一棵有效的二叉搜索树
0 <= low <= high <= 104


正解

  1. 确定终止条件
    修剪的操作并不是在终止条件上进行的,所以就是遇到空节点返回就可以了。
  2. 确定单层递归的逻辑
    如果root(当前节点)的元素小于low的数值,那么应该递归右子树,并返回右子树符合条件的头结点。
    如果root(当前节点)的元素大于high的,那么应该递归左子树,并返回左子树符合条件的头结点。
    接下来要将下一层处理完左子树的结果赋给root->left,处理完右子树的结果赋给root->right。
  3. 最后返回root节点
上代码(●'◡'●)
class Solution {
public:
    TreeNode* trimBST(TreeNode* root, int low, int high) {
        if (root == nullptr ) return nullptr;
        if (root->val < low) {
            TreeNode* right = trimBST(root->right, low, high); // 寻找符合区间[low, high]的节点
            return right;
        }
        if (root->val > high) {
            TreeNode* left = trimBST(root->left, low, high); // 寻找符合区间[low, high]的节点
            return left;
        }
        root->left = trimBST(root->left, low, high); // root->left接入符合条件的左孩子
        root->right = trimBST(root->right, low, high); // root->right接入符合条件的右孩子
        return root;
    }
};

精简:

class Solution {
public:
    TreeNode* trimBST(TreeNode* root, int low, int high) {
        if (root == nullptr) return nullptr;
        if (root->val < low) return trimBST(root->right, low, high);
        if (root->val > high) return trimBST(root->left, low, high);
        root->left = trimBST(root->left, low, high);
        root->right = trimBST(root->right, low, high);
        return root;
    }
};

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

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

示例 1:

输入:nums = [-10,-3,0,5,9]
输出:[0,-3,9,-10,null,5]
解释:[0,-10,5,null,-3,null,9] 也将被视为正确答案:

示例 2:

输入:nums = [1,3]
输出:[3,1]
解释:[1,null,3] 和 [3,1] 都是高度平衡二叉搜索树。

提示:

1 <= nums.length <= 104
-104 <= nums[i] <= 104
nums 按 严格递增 顺序排列


正解

其实数组构造二叉树,构成平衡树是自然而然的事情;
因为大家默认都是从数组中间位置取值作为节点元素,一般不会随机取。
所以想构成不平衡的二叉树是自找麻烦。
本质就是寻找分割点,分割点作为当前节点,然后递归左区间和右区间。
分割点就是数组中间位置的节点。
那么为问题来了,如果数组长度为偶数,中间节点有两个,取哪一个?
取哪一个都可以,只不过构成了不同的平衡二叉搜索树。

  1. 确定递归函数返回值及其参数
    删除二叉树节点,增加二叉树节点,都是用递归函数的返回值来完成,这样是比较方便的。
    那么本题要构造二叉树,依然用递归函数的返回值来构造中节点的左右孩子。
    再来看参数,首先是传入数组,然后就是左下标left和右下标right
    在构造二叉树的时候尽量不要重新定义左右区间数组,而是用下标来操作原数组。
  2. 确定递归终止条件
    这里定义的是左闭右闭的区间,所以当区间 left > right的时候,就是空节点了。
  3. 确定单层递归的逻辑
    首先取数组中间元素的位置,不难写出int mid = (left + right) / 2;
    这么写其实有一个问题,就是数值越界,例如left和right都是最大int,这么操作就越界了
    所以可以这么写:int mid = left + ((right - left) / 2);
    取了中间位置,就开始以中间位置的元素构造节点,代码:TreeNode* root = new TreeNode(nums[mid]);。
    接着划分区间,root的左孩子接住下一层左区间的构造节点,右孩子接住下一层右区间构造的节点。
    最后返回root
上代码(●'◡'●)
class Solution {
private:
    TreeNode* traversal(vector<int>& nums, int left, int right) {
        if (left > right) return nullptr;
        int mid = left + ((right - left) / 2);
        TreeNode* root = new TreeNode(nums[mid]);
        root->left = traversal(nums, left, mid - 1);
        root->right = traversal(nums, mid + 1, right);
        return root;
    }
public:
    TreeNode* sortedArrayToBST(vector<int>& nums) {
        TreeNode* root = traversal(nums, 0, nums.size() - 1);
        return root;
    }
};
From:https://www.cnblogs.com/Murder-sans/p/18370393/dmsxl_Day21
本文地址: http://www.shuzixingkong.net/article/1279
0评论
提交 加载更多评论
其他文章 CSS2基础(part-1)
CSS2基础 基础 简介 【全称】Cascading Style Sheets,又名层叠样式表 层叠:一层一层涂上去 表:列表 样式:如文字大小,颜色,元素宽高等。 CSS 描述了在屏幕、纸质、音频等其他媒体上的元素应该如何被渲染的问题。 语言类型 标记语言,为HTML结构美化样式,实现语义与效果的
Prometheus部署以及问题解决
Prometheus作用: Prometheus监控(Prometheus Monitoring)是一种开源的系统监控和警报工具。它最初由SoundCloud开发并于2012年发布,并在2016年加入了云原生计算基金会(CNCF)。Prometheus监控旨在收集、存储和查询各种指标数据,以帮助用户
Prometheus部署以及问题解决 Prometheus部署以及问题解决 Prometheus部署以及问题解决
.NET 智能组件完全开源
Daniel Roth在2024年3月20日发布了一篇文章: .NET 智能组件简介 – AI 驱动的 UI 控件。文章主要介绍了.NET Smart Components,这是一系列可以快速轻松地添加到.NET应用程序中的AI驱动的UI组件。这些组件旨在简化在现有软件中添加AI功能的过程,无需花费
一文讲清楚算法刷题-计算机专业新生必看
哈喽,大家好,我是Sunny,你也可以叫我萨宁,一个热爱分享编程知识的程序员。我的昵称是Sunny不要停,寓意是美好的晴朗日子不要停下来,希望大家都能每天开开心心的。我的频道主要分享编程知识,生活,大学计算机学科学习,考研经验。目前已经上岸某211计算机专业,有大学学习,考研相关的问题,欢迎关注我,
一文讲清楚算法刷题-计算机专业新生必看 一文讲清楚算法刷题-计算机专业新生必看 一文讲清楚算法刷题-计算机专业新生必看
WPF:静态、动态资源以及资源词典
WPF:静态、动态资源以及资源词典 静态资源与动态资源 我们常常会使用样式或者控件模板放在Window.Resources中,比如这样: 静态资源与动态资源使用如下: &lt;Window.Resources&gt; &lt;SolidColorBrush x:Key=&quot;SolidColo
WPF:静态、动态资源以及资源词典 WPF:静态、动态资源以及资源词典 WPF:静态、动态资源以及资源词典
RabbitMQ 基础概念与架构设计及工作机制学习总结
什么是RabbitMQ MQ全称为Message Queue,即消息队列. 它也是一个队列,遵循FIFO原则 。RabbitMQ则是一个开源的消息中间件,由erlang语言开发,基于AMQP协议实现的一个软件产品,提供应用程序之间的通信方法,在分布式系统开发中广泛应用。 AMQP协议 AMQP,即A
RabbitMQ 基础概念与架构设计及工作机制学习总结 RabbitMQ 基础概念与架构设计及工作机制学习总结 RabbitMQ 基础概念与架构设计及工作机制学习总结
Java微信授权登录小程序接口
1.微信授权登录小程序的流程是什么 微信授权登录小程序的流程是一个涉及前端和后端交互的过程,主要目的是让用户能够使用微信账号快速登录小程序,避免重复输入用户名和密码。以下是该流程的详细步骤: 1.1前端操作 (1)触发登录: 用户在小程序中点击“登录”按钮或进入需要登录的页面时,系统会自动弹出授权登
async await 状态机理解
public async Task&lt;string&gt; Wait3S() { await Task.Delay(3000); Console.WriteLine(&quot;Wait 3 S&quot;); return &quot;&quot;; } #region 异步任务-状态机 #i