代码随想录算法训练营day 23|第六章 二叉树part09

2024-02-04 11:28

本文主要是介绍代码随想录算法训练营day 23|第六章 二叉树part09,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

669. 修剪二叉搜索树 

这道题目比较难,比 添加增加和删除节点难的多,建议先看视频理解。

题目链接/文章讲解: 代码随想录

视频讲解: 你修剪的方式不对,我来给你纠正一下!| LeetCode:669. 修剪二叉搜索树_哔哩哔哩_bilibili

递归的做法是将首先确定当前节点的值是否小于左边界,如果小于左边界,那么就给它的祖先节点返回当前节点的右孩子(注意不是直接返回右孩子,而是它的返回递归函数),同理如果它的值大于右边界,就返回当前节点的左孩子,如果恰好在边界内部,那就将它的左右孩子处理一下,处理成在边界内部的,然后返回当前节点——

TreeNode* trimBST(TreeNode* root, int low, int high) {if(root==NULL) return root;TreeNode* tmp=root;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;}

文章给出的迭代做法主要分成三部分:

  1. 处理根节点,使得根节点符合要求,如果不符合就一直移动根节点,直至指向符合要求的节点
  2. 处理剩下的节点,使得满足大于左边界
  3. 处理剩下的节点,使得满足小于右边界

这道题首先处理根节点,是因为一旦根节点符合要求了,那么在处理剩下的两个步骤的时候就会简单很多,因为小于左边界的节点只能是在根节点的左边,而大于右边界的节点只能在根节点的右边。

在处理剩下两个步骤的时候,比如第二步,要使得指针一直向着左孩子移动,因为这样才能找到最小的;
由于要改变节点的指向,所以指针不能直接指向要处理的节点(也就是当前节点),而是要指向它的父母节点;
如果遇到某个节点的左孩子的值小于左边界,那就要一直执行将左孩子的右孩子赋值给当前节点的左孩子,直到使得当前节点满足要求为止,这样就能保证,当前节点以及它的右孩子一定是大于左边界的,这样再移动到当前节点的左孩子——

TreeNode* trimBST(TreeNode* root, int L, int R) {if (!root) return nullptr;// 处理头结点,让root移动到[L, R] 范围内,注意是左闭右闭while (root != nullptr && (root->val < L || root->val > R)) {if (root->val < L) root = root->right; // 小于L往右走else root = root->left; // 大于R往左走}TreeNode *cur = root;// 此时root已经在[L, R] 范围内,处理左孩子元素小于L的情况while (cur != nullptr) {while (cur->left && cur->left->val < L) {cur->left = cur->left->right;}cur = cur->left;}cur = root;// 此时root已经在[L, R] 范围内,处理右孩子大于R的情况while (cur != nullptr) {while (cur->right && cur->right->val > R) {cur->right = cur->right->left;}cur = cur->right;}return root;}

这是我自己写的迭代,和文章里面有异曲同工之处,我并没有首先处理根节点,而是直接将过程分成保证左边界符合要求和保证右边界符合要求,故而在这过程中,必然有可能存在根节点的变化,所以要及时更新根节点。

对于如何保证边界,比如保证左边界,首先判断当前节点是否越界,如果已经越界,那就需要进一步判断当前节点的父母节点是否为空,如果pre为空,就证明当前节点一定是根节点,根节点不符合要求,那就要直接更新根节点(更新为之前根节点的右孩子),而且不需要对pre进行更新,是因为修改的是根节点,而根节点的父母节点固定是NULL;
如果pre不为空,就证明不是根节点,那就要让当前节点的父母节点的左孩子指向更新为当前节点的右孩子(一定是更新的父母节点的左孩子,是因为全程都是将节点在向左移动,不会涉及到右孩子);
无论pre是否为空,也就是无论是否根节点出现问题,都要将当前节点移动到它的右孩子(不用管它的左孩子,因为左孩子一定更小,更不符合要求),因为当前节点和它的左孩子都已经被舍弃;
而如果当前节点符合要求,那么就要更新pre节点,将节点移动到当前节点的左孩子——

TreeNode* trimBST(TreeNode* root, int low, int high) {if(root==NULL) return root;TreeNode* pre=NULL;TreeNode* newRoot=root;while(root){if(root->val<low){if(!pre) {newRoot=root->right; }else pre->left=root->right;root=root->right;}else{pre=root;root=root->left; } }root=newRoot;pre=NULL;while(root){if(root->val>high){if(!pre) {newRoot=root->left;    }else {pre->right=root->left;}root=root->left;}else {pre=root;root=root->right;}}return newRoot;}

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

本题就简单一些,可以尝试先自己做做。

代码随想录

视频讲解:构造平衡二叉搜索树!| LeetCode:108.将有序数组转换为二叉搜索树_哔哩哔哩_bilibili

这道题平衡是最不需要注意的点,几乎能创建就会创建一个平衡的二叉搜索树,所以重点再利用有序数组创建搜索二叉树。

这道题使用递归很好解决,主要是注意:去两个下标的中点的话是(left+right)/2,而不是(right-left)/2——

TreeNode* buildBST(vector<int>& nums,int left,int right){if(right-left==0) return NULL;TreeNode* node=new TreeNode(nums[(right+left)/2]);if(right-left==1) return node;node->left=buildBST(nums,left,(right+left)/2);node->right=buildBST(nums,(right+left)/2+1,right);return node;}TreeNode* sortedArrayToBST(vector<int>& nums) {return buildBST(nums,0,nums.size());}

迭代法。注意文章里面这次使用的不是左闭右开区间而是左闭右闭区间。这道题无关出队的先后顺序,只要每次三个容器的对应位置的内容是相互匹配的就行,所以也可以用三个栈来实现,大差不差。用容器来储存左右区间下标,因为左右区间下标不能实现同时更新,而且更新的顺序不固定,所以需要使用两个容器来储存。

每次取出要遍历的一个节点,给它赋值(通过它的左右区间下标来确定它的值,它的值一定是区间中点处的值),然后考虑当前节点是否存在左右孩子(通过左右区间下标和中点下标来判断),如果存在左右孩子,那就先将一个无效节点储存进去(这时候的值是没有意义的,需要在弹出的时候重新赋值),然后更新相应的区间下标——

TreeNode* sortedArrayToBST(vector<int>& nums) {if (nums.size() == 0) return nullptr;TreeNode* root = new TreeNode(0);   // 初始根节点queue<TreeNode*> nodeQue;           // 放遍历的节点queue<int> leftQue;                 // 保存左区间下标queue<int> rightQue;                // 保存右区间下标nodeQue.push(root);                 // 根节点入队列leftQue.push(0);                    // 0为左区间下标初始位置rightQue.push(nums.size() - 1);     // nums.size() - 1为右区间下标初始位置while (!nodeQue.empty()) {TreeNode* curNode = nodeQue.front();nodeQue.pop();int left = leftQue.front(); leftQue.pop();int right = rightQue.front(); rightQue.pop();int mid = left + ((right - left) / 2);curNode->val = nums[mid];       // 将mid对应的元素给中间节点if (left <= mid - 1) {          // 处理左区间curNode->left = new TreeNode(0);nodeQue.push(curNode->left);leftQue.push(left);rightQue.push(mid - 1);}if (right >= mid + 1) {         // 处理右区间curNode->right = new TreeNode(0);nodeQue.push(curNode->right);leftQue.push(mid + 1);rightQue.push(right);}}return root;}

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

本题也不难,在 求二叉搜索树的最小绝对差 和 众数 那两道题目 都讲过了 双指针法,思路是一样的。

代码随想录

视频讲解:

普大喜奔!二叉树章节已全部更完啦!| LeetCode:538.把二叉搜索树转换为累加树_哔哩哔哩_bilibili

这道题其实就是中序遍历(不过是改成了右中左)的时候修改一下遍历节点的值,所以只需要在遍历模板的基础上修改一下即可,迭代法也是同理——

class Solution {
private:int pre = 0; // 记录前一个节点的数值void traversal(TreeNode* cur) { // 右中左遍历if (cur == NULL) return;traversal(cur->right);cur->val += pre;pre = cur->val;traversal(cur->left);}
public:TreeNode* convertBST(TreeNode* root) {pre = 0;traversal(root);return root;}
};

总结篇  

好了,二叉树大家就这样刷完了,做一个总结吧

代码随想录

注意——

  • 涉及到二叉树的构造,无论普通二叉树还是二叉搜索树一定前序,都是先构造中节点。

  • 求普通二叉树的属性,一般是后序,一般要通过递归函数的返回值做计算。

  • 求二叉搜索树的属性,一定是中序了,要不白瞎了有序性了。

 

这篇关于代码随想录算法训练营day 23|第六章 二叉树part09的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

python使用Akshare与Streamlit实现股票估值分析教程(图文代码)

《python使用Akshare与Streamlit实现股票估值分析教程(图文代码)》入职测试中的一道题,要求:从Akshare下载某一个股票近十年的财务报表包括,资产负债表,利润表,现金流量表,保存... 目录一、前言二、核心知识点梳理1、Akshare数据获取2、Pandas数据处理3、Matplotl

Django开发时如何避免频繁发送短信验证码(python图文代码)

《Django开发时如何避免频繁发送短信验证码(python图文代码)》Django开发时,为防止频繁发送验证码,后端需用Redis限制请求频率,结合管道技术提升效率,通过生产者消费者模式解耦业务逻辑... 目录避免频繁发送 验证码1. www.chinasem.cn避免频繁发送 验证码逻辑分析2. 避免频繁

精选20个好玩又实用的的Python实战项目(有图文代码)

《精选20个好玩又实用的的Python实战项目(有图文代码)》文章介绍了20个实用Python项目,涵盖游戏开发、工具应用、图像处理、机器学习等,使用Tkinter、PIL、OpenCV、Kivy等库... 目录① 猜字游戏② 闹钟③ 骰子模拟器④ 二维码⑤ 语言检测⑥ 加密和解密⑦ URL缩短⑧ 音乐播放

Python使用Tenacity一行代码实现自动重试详解

《Python使用Tenacity一行代码实现自动重试详解》tenacity是一个专为Python设计的通用重试库,它的核心理念就是用简单、清晰的方式,为任何可能失败的操作添加重试能力,下面我们就来看... 目录一切始于一个简单的 API 调用Tenacity 入门:一行代码实现优雅重试精细控制:让重试按我

Python实现MQTT通信的示例代码

《Python实现MQTT通信的示例代码》本文主要介绍了Python实现MQTT通信的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录1. 安装paho-mqtt库‌2. 搭建MQTT代理服务器(Broker)‌‌3. pytho

MySQL进行数据库审计的详细步骤和示例代码

《MySQL进行数据库审计的详细步骤和示例代码》数据库审计通过触发器、内置功能及第三方工具记录和监控数据库活动,确保安全、完整与合规,Java代码实现自动化日志记录,整合分析系统提升监控效率,本文给大... 目录一、数据库审计的基本概念二、使用触发器进行数据库审计1. 创建审计表2. 创建触发器三、Java

Java中的雪花算法Snowflake解析与实践技巧

《Java中的雪花算法Snowflake解析与实践技巧》本文解析了雪花算法的原理、Java实现及生产实践,涵盖ID结构、位运算技巧、时钟回拨处理、WorkerId分配等关键点,并探讨了百度UidGen... 目录一、雪花算法核心原理1.1 算法起源1.2 ID结构详解1.3 核心特性二、Java实现解析2.

Java中调用数据库存储过程的示例代码

《Java中调用数据库存储过程的示例代码》本文介绍Java通过JDBC调用数据库存储过程的方法,涵盖参数类型、执行步骤及数据库差异,需注意异常处理与资源管理,以优化性能并实现复杂业务逻辑,感兴趣的朋友... 目录一、存储过程概述二、Java调用存储过程的基本javascript步骤三、Java调用存储过程示

Visual Studio 2022 编译C++20代码的图文步骤

《VisualStudio2022编译C++20代码的图文步骤》在VisualStudio中启用C++20import功能,需设置语言标准为ISOC++20,开启扫描源查找模块依赖及实验性标... 默认创建Visual Studio桌面控制台项目代码包含C++20的import方法。右键项目的属性:

MySQL数据库的内嵌函数和联合查询实例代码

《MySQL数据库的内嵌函数和联合查询实例代码》联合查询是一种将多个查询结果组合在一起的方法,通常使用UNION、UNIONALL、INTERSECT和EXCEPT关键字,下面:本文主要介绍MyS... 目录一.数据库的内嵌函数1.1聚合函数COUNT([DISTINCT] expr)SUM([DISTIN