代码随想录算法训练营Day22|235.二叉搜索树的最近公共祖先、701.二叉搜索树中的插入操作、450.删除二叉搜索树中的节点

二叉搜索树的最近公共祖先

不考虑二叉搜索树这一条件的话,普通的二叉搜索树搜索最近的公共祖先就是昨日的做法,这种做法也能解决二叉搜索树的最近公共祖先。

class Solution {
public:TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {// 如果当前节点为空,或者等于p或q,直接返回当前节点if (root == nullptr || root == p || root == q) {return root;}// 在左右子树中递归寻找p和qTreeNode* left = lowestCommonAncestor(root->left, p, q);TreeNode* right = lowestCommonAncestor(root->right, p, q);// 如果左右子树的返回值都不为空,说明当前节点就是最近公共祖先if (left != nullptr && right != nullptr) {return root;}// 否则,返回非空的子树返回值return left != nullptr ? left : right;}
};

没有用上二叉搜索树这一条件,但也能解题,但效率较低。

针对二叉搜索树,我们之前有做过,当对二叉搜索树进行中序编历时,结果是一个递增的数组。即公共祖先,val值必定处于p和q之间。

当从根节点向下遍历的过程中,如果遇到节点val值位于p和q之间,那么就寻找到了最近的公共祖先。具体参考代码随想录B站视频。

二叉搜索树找祖先就有点不一样了!| 235. 二叉搜索树的最近公共祖先_哔哩哔哩_bilibiliicon-default.png?t=N7T8https://www.bilibili.com/video/BV1Zt4y1F7ww/?spm_id_from=333.788&vd_source=fc4a6e70e3a87b7ea67c2024e326e7c5从上到下遍历,考虑层序遍历的方式,具体代码如下:

class Solution {
public:TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {queue<TreeNode*>queue;if(root == nullptr){return nullptr;}queue.push(root);TreeNode*cur = root;int min = p->val>q->val?q->val:p->val;//找到p,q中val的较小值int max = p->val<q->val?q->val:p->val;//找到p,q中val的较大值while(!queue.empty()){//层序遍历过程int level_size = queue.size();for(int i = 0; i <level_size; i ++){cur = queue.front();queue.pop();if(cur->val<=max and cur->val>=min){return cur;}//找到节点属于p,q间则返回if(cur->left)queue.push(cur->left);if(cur->right)queue.push(cur->right);} }return nullptr;}
};

算法的时间复杂度和空间复杂度均为O(n)。

前序递归,中左右,参考代码随想录

代码随想录 (programmercarl.com)icon-default.png?t=N7T8https://programmercarl.com/0235.%E4%BA%8C%E5%8F%89%E6%90%9C%E7%B4%A2%E6%A0%91%E7%9A%84%E6%9C%80%E8%BF%91%E5%85%AC%E5%85%B1%E7%A5%96%E5%85%88.html#%E6%80%9D%E8%B7%AF

class Solution {
private:TreeNode* traversal(TreeNode* cur, TreeNode* p, TreeNode* q) {if (cur == NULL) return cur;// 中if (cur->val > p->val && cur->val > q->val) {   // 左TreeNode* left = traversal(cur->left, p, q);if (left != NULL) {return left;}}if (cur->val < p->val && cur->val < q->val) {   // 右TreeNode* right = traversal(cur->right, p, q);if (right != NULL) {return right;}}return cur;}
public:TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {return traversal(root, p, q);}
};

二叉搜索树中的插入操作

由于树中的节点值是独一无二的,在二叉搜索树中寻找值应插入的节点的父节点位置,然后创建一个新节点并将其插入即可。

代码整体如下

class Solution {
public:// 在二叉搜索树中查找适合插入新节点的位置,并返回该位置的父节点TreeNode* findBST(TreeNode* root, int val) {// 如果当前节点为空,返回nullptr,表示没有找到合适的插入位置if (root == nullptr) return nullptr;// 如果val小于当前节点的值,应该在左子树中继续查找if (val < root->val) {// 如果左子节点为空,当前位置就是适合插入新节点的位置if (root->left == nullptr) return root;// 否则,在左子树中继续查找return findBST(root->left, val);} else {// 如果val大于或等于当前节点的值,应该在右子树中继续查找// 如果右子节点为空,当前位置就是适合插入新节点的位置if (root->right == nullptr) return root;// 否则,在右子树中继续查找return findBST(root->right, val);}}// 在二叉搜索树中插入一个新的节点TreeNode* insertIntoBST(TreeNode* root, int val) {// 创建新节点TreeNode* newnode = new TreeNode(val);// 如果树为空,新节点即为根节点if (root == nullptr) {return newnode;}// 查找适合插入新节点的位置,并得到该位置的父节点TreeNode* parent = findBST(root, val);// 根据val的值决定新节点是作为左子节点还是右子节点if (val < parent->val) {parent->left = newnode;} else {parent->right = newnode;}// 返回根节点return root;}
};

算法的时间复杂度为O(logn)(二叉搜索树,最差为O(n)),空间复杂度为O(1)。

删除二叉搜索树中的节点

有五种情况

1.未找到需要删除的节点

2.删除的是叶节点

3.删除的节点仅有右子树

4.删除的节点仅有左子树

5.删除的节点左右子树都在

针对这五种情况,有以下五种解答方案

1.直接返回二叉搜索树的根节点

2.直接删除即可

3.将右孩子代替被删除的节点

4.将左孩子代替被删除的节点

5.有两种解决方式,分别是让被删除节点的右孩子或左孩子即位,以右孩子即位,查找右子树下的最小值节点,将节点的左子树全部接入该节点,或以左孩子即位,查找左子树下的最大值节点,将节点的右子树全部接入该节点。

class Solution {
public:TreeNode* find_parent(TreeNode* root, int key) {if (root == nullptr || (root->left == nullptr && root->right == nullptr)) {return nullptr; // 如果当前节点为空或为叶节点,返回nullptr}if (root->left != nullptr && root->left->val == key) {return root; // 如果左子节点的值等于key,返回当前节点作为父节点}if (root->right != nullptr && root->right->val == key) {return root; // 如果右子节点的值等于key,返回当前节点作为父节点}if (key < root->val) {return find_parent(root->left, key); // 如果key小于当前节点的值,递归地在左子树中查找} else {return find_parent(root->right, key); // 如果key大于或等于当前节点的值,递归地在右子树中查找}}TreeNode* deleteNode(TreeNode* root, int key) {if (root == nullptr) return nullptr;if (root->val == key) {// 要删除的节点是根节点if (root->left == nullptr) return root->right; // 只有右子树if (root->right == nullptr) return root->left; // 只有左子树// 有两个子节点,找到右子树的最小节点TreeNode* minNode = getMin(root->right);root->val = minNode->val; // 将右子树的最小节点的值赋给当前节点root->right = deleteNode(root->right, minNode->val); // 删除右子树中的最小节点} else if (key < root->val) {root->left = deleteNode(root->left, key); // 在左子树中递归删除} else {root->right = deleteNode(root->right, key); // 在右子树中递归删除}return root;}TreeNode* getMin(TreeNode* node) {while (node->left != nullptr) node = node->left;return node;}
};

算法的时间复杂度需要考虑树的高度,查找树中要删除的节点,需要O(logn)的时间,当需要删除的节点有两个子节点时,需要找到右子树中的最小节点,这同样需要O(logn)的时间,最坏情况下时间复杂度为O(logn)。

空间复杂度主要取决于递归栈的深度,因此,空间复杂度也为O(logn)。

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

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

相关文章

STM32读写内部FLASH读取芯片id

文章目录 读写内部Flash接线程序编写测试效果补充 读取芯片id代码编写 读写内部Flash 接线 程序编写 首先使用ThisFlash.c来写入flash的基本操作&#xff0c;写入、读取、擦除&#xff0c;然后使用Store.c配合数组来进行主存与flash的交互 ThisFlash.c #include "stm32…

为什么工控现场会用到Profinet转Modbus网关设备

一、背景&#xff1a; 工控现场之所以需要使用Profinet转Modbus网关&#xff0c;是因为工控系统中常常存在不同厂家设备之间通讯协议不一致的问题。而Modbus和Profinet分别代表着两种不同的通信协议&#xff0c;Profinet通常用于较新的设备&#xff0c;而Modbus则是比较老的通…

ch2应用层--计算机网络期末复习

2.1应用层协议原理 网络应用程序位于应用层 开发网络应用程序: 写出能够在不同的端系统上通过网络彼此通信的程序 2.1.1网络应用程序体系结构分类: 客户机/服务器结构 服务器: 总是打开(always-on)具有固定的、众所周知的IP地址 主机群集常被用于创建强大的虚拟服务器 客…

基于51单片机的温控风扇的设计–仿真设计

可实现通过DS18B20测量当前环境温度 可实现通过温度自动控制风扇转速 可实现通过按键设置不同风速对应的温度 可实现通过按键切换自动、手动模式 可实现在手动模式下通过按键调整风扇转速 可实现通过LCD1602显示温度、风扇转速挡位、自动/手动模式

【C++】模拟实现string类

&#x1f984;个人主页:修修修也 &#x1f38f;所属专栏:C ⚙️操作环境:Visual Studio 2022 目录 一.了解项目功能 二.逐步实现项目功能模块及其逻辑详解 &#x1f38f;构建成员变量 &#x1f38f;实现string类默认成员函数 &#x1f4cc;构造函数 &#x1f4cc;析构函数…

k8s 全面掌控日志系统

概述 为了提高系统运维和故障排查的效率&#xff0c; 日志系统采用 ELK&#xff08;Elasticsearch、Logstash、Kibana&#xff09;技术栈&#xff0c;通过 FileBeats 作为日志收集器&#xff0c;将来自不同节点的日志数据汇总并存储在 Elasticsearch 中&#xff0c;最终通过 K…

创建一个新的Spring Security应用程序,并使用JDBC连接数据库

创建一个新的Spring Security应用程序&#xff0c;并使用JDBC连接数据库 在这个教程中&#xff0c;我们将学习如何创建一个新的Spring Security应用程序&#xff0c;使用JDBC连接数据库以获取用户信息并进行认证。我们还将学习如何配置Spring Security以从数据库中获取用户和权…

Vue3使用Composition API实现响应式

title: Vue3使用Composition API实现响应式 date: 2024/5/29 下午8:10:24 updated: 2024/5/29 下午8:10:24 categories: 前端开发 tags: Vue3CompositionRefsReactiveWatchLifecycleDebugging 1. 介绍 Composition API是Vue.js 3中新增的一组API&#xff0c;用于在组件中组…

Python 之微信指数小程序数据抓取

Fiddler安装和设置 安装 Fiddler 安装包可以从这里获取&#xff0c;如果失效了可以自己网上找一个安装。 链接&#xff1a;https://pan.baidu.com/s/1N30BoDWm2_dBL8i8GRzK5g?pwd1znv 提取码&#xff1a;1znv 然后就是点击安装就好了&#xff0c;没什么好多说的。 启用…

刷代码随想录有感(83):贪心算法——最大子数组和

题干&#xff1a; 代码&#xff1a; class Solution { public:int maxSubArray(vector<int>& nums) {int res INT_MIN;int count 0;for(int i 0; i < nums.size(); i){count nums[i];if(count > res) res count;if(count < 0)count 0;}return res;} …

【创作活动】探索 GPT-4o:下一代语言模型的技术革命

&#x1f604; 19年之后由于某些原因断更了三年&#xff0c;23年重新扬帆起航&#xff0c;推出更多优质博文&#xff0c;希望大家多多支持&#xff5e; &#x1f337; 古之立大事者&#xff0c;不惟有超世之才&#xff0c;亦必有坚忍不拔之志 &#x1f390; 个人CSND主页——Mi…

HTTP报文

HTTP报文 报文流 HTTP报文是在HTTP引用程序之间发送的数据块&#xff0c;这些数据块以一种文本形式的元信息开头&#xff0c;这些信息描述了报文的内容和含义&#xff0c;后面跟着可选的数据部分&#xff0c;这些报文在客户端&#xff0c;服务器和代理之间流动。 报文流入源…

前端项目开发,3个HTTP请求工具

这一小节&#xff0c;我们介绍一下前端项目开发中&#xff0c;HTTP请求会用到的3个工具&#xff0c;分别是fetch、axios和js-tool-big-box中的jsonp请求。那么他们都有哪些小区别呢&#xff1f;我们一起来看一下。 目录 1 fetch 2 axios 3 js-tool-big-box 的 jsonp 请求 …

拷贝构造、移动构造、拷贝赋值、移动赋值

最近在学习C的拷贝构造函数时发现一个问题&#xff1a;在函数中返回局部的类对象时&#xff0c;并没有调用拷贝构造函数。针对这个问题&#xff0c;查阅了一些资料&#xff0c;这里记录整理一下。 调用拷贝构造函数的三种情况&#xff1a; ① 用一个类去初始化另一个对象时&a…

【数据结构与算法 | 基础篇 | 队列篇】力扣102, 107

1. 力扣102 : 二叉树的层序遍历 (1). 题 给你二叉树的根节点 root &#xff0c;返回其节点值的 层序遍历 。 &#xff08;即逐层地&#xff0c;从左到右访问所有节点&#xff09;。 示例 1&#xff1a; 输入&#xff1a;root [3,9,20,null,null,15,7] 输出&#xff1a;[[3]…

刷爆leetcode第六期

题目一 用队列实现栈 请你仅使用两个队列实现一个后入先出&#xff08;LIFO&#xff09;的栈&#xff0c;并支持普通栈的全部四种操作&#xff08;push、top、pop 和 empty&#xff09;。 实现 MyStack 类&#xff1a; void push(int x) 将元素 x 压入栈顶。 int pop() 移除…

【漏洞复现】大华智能物联综合管理平台 fastjson远程代码执行漏洞

0x01 产品简介 大华ICC智能物联综合管理平台对技术组件进行模块化和松耦合&#xff0c;将解决方案分层分级&#xff0c;提高面向智慧物联的数据接入与生态合作能力。 0x02 漏洞概述 由于大华智能物联综合管理平台使用了存在漏洞的Fastson组件,未经身份验让的攻击者可利用 /e…

M功能-支付平台(六)

target&#xff1a;离开柬埔寨倒计时-217day 今天突然发现我在csdn居然把我ip属地搞出来了&#xff0c;之前都没注意到&#xff0c;哎 前言 M功能演示版本做到后期(也就是第二周的后面3天)真的很心酸&#xff0c;这边安排的4后端后面都放弃了&#xff0c;觉得做不出来&#…

ARM-V9 RME(Realm Management Extension)系统架构之系统能力的内存隔离和保护

安全之安全(security)博客目录导读 目录 一、内存隔离和保护 1、颗粒PAS过滤Granular PAS filtering 2、Cache的一致性维护 2.1 物理别名点 Point of Physical Aliasing (PoPA) 2.2 加密点 3、内存(DRAM)保护 3.1 内存加密和完整性 3.2 DRAM scrubbing 本博客探讨 RME…

网络编程 —— Http使用httpClient实现页面爬虫

先去找类型的a标签 取出图片所在网址 取出https://desk.3gbizhi.com/deskMV/438.html 搭建Form界面 Http类 public static HttpClient Client { get; } static Http() {HttpClientHandler handler new HttpClientHandler();//处理消息对象//ServerCertificateCustomValidat…