二叉搜索树题目:把二叉搜索树转换为累加树

news2025/4/21 15:20:11

文章目录

  • 题目
    • 标题和出处
    • 难度
    • 题目描述
      • 要求
      • 示例
      • 数据范围
  • 前言
  • 解法一
    • 思路和算法
    • 代码
    • 复杂度分析
  • 解法二
    • 思路和算法
    • 代码
    • 复杂度分析
  • 解法三
    • 思路和算法
    • 代码
    • 复杂度分析

题目

标题和出处

标题:把二叉搜索树转换为累加树

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

难度

4 级

题目描述

要求

给定二叉搜索树的根结点 root \texttt{root} root,将其转换为累加树,原二叉搜索树中的每个结点值都变成原结点值加上比该结点值大的所有结点值总和。

二叉搜索树满足下列约束条件:

  • 结点的左子树仅包含值小于结点值的结点。
  • 结点的右子树仅包含值大于结点值的结点。
  • 左右子树也必须是二叉搜索树。

示例

示例 1:

示例 1

输入: root   =   [4,1,6,0,2,5,7,null,null,null,3,null,null,null,8] \texttt{root = [4,1,6,0,2,5,7,null,null,null,3,null,null,null,8]} root = [4,1,6,0,2,5,7,null,null,null,3,null,null,null,8]
输出: [30,36,21,36,35,26,15,null,null,null,33,null,null,null,8] \texttt{[30,36,21,36,35,26,15,null,null,null,33,null,null,null,8]} [30,36,21,36,35,26,15,null,null,null,33,null,null,null,8]

示例 2:

输入: root   =   [0,null,1] \texttt{root = [0,null,1]} root = [0,null,1]
输出: [1,null,1] \texttt{[1,null,1]} [1,null,1]

数据范围

  • 树中结点数目在范围 [0,   10 4 ] \texttt{[0, 10}^\texttt{4}\texttt{]} [0, 104]
  • -10 4 ≤ Node.val ≤ 10 4 \texttt{-10}^\texttt{4} \le \texttt{Node.val} \le \texttt{10}^\texttt{4} -104Node.val104
  • 树中的所有值各不相同
  • root \texttt{root} root 保证是有效的二叉搜索树

前言

考虑二叉搜索树的中序遍历。常规的中序遍历方法是依次遍历左子树、根结点和右子树,对于左子树和右子树使用同样的方法遍历,此时二叉搜索树的中序遍历序列是单调递增的。将中序遍历的顺序改变,依次遍历右子树、根结点和左子树,则是反向中序遍历,二叉搜索树的反向中序遍历序列是单调递减的。在反向中序遍历的过程中,对于每个结点,比该结点值大的所有结点都在该结点之前被访问。

由于对二叉搜索树反向中序遍历访问结点的顺序为结点值单调递减的顺序,因此可以通过反向中序遍历将二叉搜索树转换为累加树。反向中序遍历的过程中,维护已经访问过的结点值总和,对于访问到的每个结点,将总和加到当前结点值,则当前结点值变为该结点在累加树中对应的结点值,然后将当前结点值作为更新后的结点值总和,继续遍历其余的结点,直到遍历结束时,即可得到累加树。

解法一

思路和算法

由于二叉搜索树的反向中序遍历具有递归的性质,因此可以使用递归的方法实现中序遍历。使用递归实现反向中序遍历的做法是依次访问右子树、根结点和左子树,对于右子树和左子树使用同样的方法访问。

遍历过程中维护已经访问过的结点值总和,对于访问到的每个结点,更新结点值,并更新已经访问过的结点值总和。

代码

class Solution {
    int prev = 0;

    public TreeNode convertBST(TreeNode root) {
        if (root == null) {
            return root;
        }
        convertBST(root.right);
        root.val += prev;
        prev = root.val;
        convertBST(root.left);
        return root;
    }
}

复杂度分析

  • 时间复杂度: O ( n ) O(n) O(n),其中 n n n 是二叉搜索树的结点数。每个结点都被访问一次。

  • 空间复杂度: O ( n ) O(n) O(n),其中 n n n 是二叉搜索树的结点数。空间复杂度主要是递归调用的栈空间,取决于二叉搜索树的高度,最坏情况下二叉搜索树的高度是 O ( n ) O(n) O(n)

解法二

思路和算法

使用迭代实现反向中序遍历的做法是使用栈存储结点。

遍历过程中维护已经访问过的结点值总和,对于访问到的每个结点,更新结点值,并更新已经访问过的结点值总和。

代码

class Solution {
    public TreeNode convertBST(TreeNode root) {
        int prev = 0;
        Deque<TreeNode> stack = new ArrayDeque<TreeNode>();
        TreeNode node = root;
        while (!stack.isEmpty() || node != null) {
            while (node != null) {
                stack.push(node);
                node = node.right;
            }
            node = stack.pop();
            node.val += prev;
            prev = node.val;
            node = node.left;
        }
        return root;
    }
}

复杂度分析

  • 时间复杂度: O ( n ) O(n) O(n),其中 n n n 是二叉搜索树的结点数。每个结点都被访问一次。

  • 空间复杂度: O ( n ) O(n) O(n),其中 n n n 是二叉搜索树的结点数。空间复杂度主要是栈空间,取决于二叉搜索树的高度,最坏情况下二叉搜索树的高度是 O ( n ) O(n) O(n)

解法三

思路和算法

使用莫里斯遍历实现反向中序遍历时,由于遍历顺序是依次遍历右子树、根结点和左子树,因此和常规的莫里斯中序遍历的方向相反。对于每个结点,判断其右子树是否为空,如果当前结点的右子树不为空,则应该找到当前结点的后继结点,即当前结点的右子树中的最左边的结点,对后继结点执行相应的操作。

遍历过程中维护已经访问过的结点值总和,对于访问到的每个结点,更新结点值,并更新已经访问过的结点值总和。

代码

class Solution {
    public TreeNode convertBST(TreeNode root) {
        int prev = 0;
        TreeNode node = root;
        while (node != null) {
            if (node.right == null) {
                node.val += prev;
                prev = node.val;
                node = node.left;
            } else {
                TreeNode successor = node.right;
                while (successor.left != null && successor.left != node) {
                    successor = successor.left;
                }
                if (successor.left == null) {
                    successor.left = node;
                    node = node.right;
                } else {
                    successor.left = null;
                    node.val += prev;
                    prev = node.val;
                    node = node.left;
                }
            }
        }
        return root;
    }
}

复杂度分析

  • 时间复杂度: O ( n ) O(n) O(n),其中 n n n 是二叉搜索树的结点数。使用莫里斯遍历,每个结点最多被访问两次。

  • 空间复杂度: O ( 1 ) O(1) O(1)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1477867.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Vmware Esxi 部署Mac OS虚拟机

Vmware Esxi在创建虚拟机的时候是有Mac OS选项的&#xff0c;但是实际创建时&#xff0c;选择ISO开机后一直反复引导&#xff0c;是有问题的&#xff0c;原因是需要先解锁&#xff0c;需要在ESXI主机上修改配置并重启。 首先找到管理-服务-TSM-ssh&#xff0c;点击启动&#x…

并查集基础,死去的回忆突然攻击我

并查集普及【模板】并查集 - 洛谷 #include<iostream> #include<cstring> #include<queue> #include<vector> #include<algorithm> using namespace std; #define int long long typedef pair<int,int> PII; #define xx first #define y…

【已解决】c++如何从0配置ffmpeg

本博文源于笔者正在做的任务&#xff0c;将视频mp4每帧转换成jpg格式&#xff0c;前提是需要配置ffmpeg。因此本文从0下载教读者如何配置ffmpeg。 文章目录 1、百度搜索官网2、配置项目3、创建main.cpp测试库与包4、 总结 1、百度搜索官网 官网已经搜索到&#xff0c;我们要找…

ulimit命令

ulimit命令 ulimit 命令用于查看和设置 shell 运行时的资源限制。它可以控制各种资源&#xff0c;如文件打开数量、堆栈大小、CPU 时间等。ulimit 命令通常用于限制 shell 启动的进程的资源使用量&#xff0c;以防止系统资源被耗尽。ulimit命令的主要作用是提高系统的性能和稳…

【C++】类和对象之常引用与运算符重载

个人主页 &#xff1a; zxctscl 文章封面来自&#xff1a;艺术家–贤海林 如有转载请先通知 文章目录 1. 前言2. 常引用3. 运算符重载 1. 前言 在之前的文章中提到了常引用&#xff0c;再来看运算符重载之前&#xff0c;先来看看常引用的使用。 2. 常引用 在使用引用的时候可…

西门子WINCC客户端提示“未找到启动画面”

WinCC客户端提示“未找到启动画面”&#xff1f; 在WinCC客户机服务器架构中&#xff0c;多用户客户机互连服务器项目时&#xff0c;有时会弹出“未找到启动画面”的提示对话框&#xff0c;并且选择画面后继续弹出该对话框&#xff0c;无法正常显示画面。 可能的原因有&#x…

DevEco Studio下载与安装(Windows)

下载地址&#xff1a; HUAWEI DevEco Studio和SDK下载和升级 | HarmonyOS开发者 安装时直接点击 next 即可。 运⾏已安装的DevEco Studio&#xff0c;⾸次使⽤&#xff0c;请选择Do not import settings&#xff0c;单击OK。 1.安装Node.js 如果本地有下载&#xff0c;可以…

利用观测云实现 Kubernetes 多集群可观测

简介 观测云的工作空间接入多个 Kubernetes 集群时&#xff0c;是如何区分不同集群&#xff0c;达到多集群的可观测性&#xff1f; 增加Tag NAMESPACE&#xff1a;DataKit 选举空间&#xff0c;需要设置 ENV_NAMESPACE 环境变量&#xff0c;值为非空字符&#xff0c;不同集群…

稀疏卷积Sparse Convolution

1. 为什么提出稀疏卷积&#xff1f;它有什么好处&#xff1f; 卷积神经网络已经被证明对于二维图像信号处理是非常有效的。然而&#xff0c;对于三维点云信号&#xff0c;额外的维度显著增加了计算量。 另一方面&#xff0c;与普通图像每个像素都有值不同的是&#xff0c;一般…

从0开始python学习-53.python中flask创建简单接口

目录 1. 创建一个简单的请求,没有写方法时默认为get 2. 创建一个get请求 3. 创建一个post请求&#xff0c;默认可以使用params和表单传参 4. 带有参数的post请求 1. 创建一个简单的请求,没有写方法时默认为get from flask import Flask, request# 初始化一个flask的对象 ap…

细菌,支原体,真菌--细胞培养的最大威胁及解决方法

在科研实验中&#xff0c;微生物污染是细胞培养过程中面临的一项长期的严重威胁。侵入的支原体、细菌、酵母菌和真菌可以直接杀死您宝贵的细胞或者彻底改变培养细胞的特性&#xff0c;导致不准确的实验结果&#xff0c;既耗费时间又浪费资源。InvivoGen/欣博盛生物可以提供多种…

QT Mingw编译ffmpeg源码以及测试

文章目录 前言下载msys2ysamFFmpeg 搭建编译环境安装msys2安装QT Mingw编译器到msys环境中安装ysam测试 编译FFmpeg 前言 FFmpeg不像VLC有支持QT的库文件&#xff0c;它仅提供源码&#xff0c;需要使用者自行编译成对应的库&#xff0c;当使用QTFFmpeg实现播放视频以及视频流时…

Linux系统安装

Linux系统安装 安装包链接 链接&#xff1a;https://pan.baidu.com/s/1FdP7TH90UvKUQuiL2yeGCA 提取码&#xff1a;c49n安装包内容 虚拟机执行文件 详细安装教程 虚拟机密钥 Ubuntu 安装步骤 先点击虚拟机的.EXE文件安装&#xff0c;打开安装教程&#xff0c;有详细的说明。

【Acwing】154滑动窗口

#include<bits/stdc.h> using namespace std;int n,k; int nums[1000010] {0};int main() {ios::sync_with_stdio(0);cin.tie(0), cout.tie(0);cin>>n>>k;// 下面求最小值int cnt 0;int res[1000010] {0};deque<int> q; // 双端队列 for(int i0;i&l…

Zoho Bigin斩获PCMag推崇:小企业首选CRM软件

当我们谈论企业怎样选择CRM管理系统时&#xff0c;大型企业、中型企业以及小型或初创等不同规模的企业需求各不相同&#xff0c;很难用一套软件来适配所有公司。以小企业为例&#xff0c;大多面临预算紧张、没有专业的IT部门或支持团队、暂时用不到高级定制功能等现状。基于这个…

R语言提高效率的技巧(2)

1、R project&#xff1a;加载数据和脚本不需要设置文件夹&#xff1b;打包project文件夹后发给别人&#xff0c;在另外一个环境中&#xff0c;点击.Rproj文件即可运行。 2、下载原始R包&#xff0c;进行补充和修改。 https://cran.r-project.org/web/packages/available_pack…

智能分析网关V4电瓶车检测与烟火算法,全面提升小区消防安全水平

2024年2月23日&#xff0c;南京市某小区因电瓶车停放处起火引发火灾事故&#xff0c;造成巨大人员伤亡和损失。根据国家消防救援局的统计&#xff0c;2023年全国共接报电动自行车火灾2.1万起。电瓶车火灾事故频发&#xff0c;这不得不引起我们的重视和思考&#xff0c;尤其是在…

RK3568 android11 调试陀螺仪模块 MPU6500

一&#xff0c;MPU6500功能介绍 1.简介 MPU6500是一款由TDK生产的运动/惯性传感器&#xff0c;属于惯性测量设备&#xff08;IMU&#xff09;的一种。MPU6500集成了3轴加速度计、3轴陀螺仪和一个板载数字运动处理器&#xff08;DMP&#xff09;&#xff0c;能够提供6轴的运动…

Socket网络编程(三)——TCP快速入门

目录 概述TCP连接可靠性1. 三次握手过程2. 四次挥手过程3. 为什么挥手需要四次&#xff1f; 传输可靠性TCP核心APITCP传输初始化配置&建立连接客户端创建Socket建立连接服务端创建ServerSocket监听连接ServerSocket 和 Socket的关系 Socket基本数据类型传输客户端数据传输服…

【MySQL】内置函数 -- 详解

一、日期函数 日期&#xff1a;年月日时间&#xff1a;时分秒 1、获得年月日 2、获得时分秒 3、获得时间戳 4、在日期的基础上加日期 5、在日期的基础上减去时间 6、计算两个日期之间相差多少天 7、获得当前时间 ⚪练习 &#xff08;1&#xff09;记录生日 &#xff08;2&…