代码随想录算法训练营第二十一天| LeetCode 530.二叉搜索树的最小绝对差 、501.二叉搜索树中的众数、236. 二叉树的最近公共祖先

本文主要是介绍代码随想录算法训练营第二十一天| LeetCode 530.二叉搜索树的最小绝对差 、501.二叉搜索树中的众数、236. 二叉树的最近公共祖先,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、LeetCode 530.二叉搜索树的最小绝对差

题目链接/文章讲解/视频讲解: https://programmercarl.com/0530.%E4%BA%8C%E5%8F%89%E6%90%9C%E7%B4%A2%E6%A0%91%E7%9A%84%E6%9C%80%E5%B0%8F%E7%BB%9D%E5%AF%B9%E5%B7%AE.html
状态:已解决

1.思路

        这道题与LeetCode98很相似,只是它需要保证前一个节点值小于后一个节点值,而这里则是求二者差值中的最小值,二者本质都是查看中序遍历的情况下,相邻节点的值。其实昨天那个题也可以用双指针来做,但是由于稍微难理解就丢弃了那个做法,现在发现躲来躲去还是躲不掉。

二叉搜索树中的双指针法:

        按照中序遍历的顺序(二叉搜索树一定是中序遍历,否则无序),prr指针指向前一个节点,cur指针指向后一个节点。然后每次递归一个节点后,将cur的值赋给pre,cur再继续递归下一个节点,那么相当于下轮递归子树时,cur指向的节点始终为pre节点的下一个节点,即二者同步移动,每次都相差一个节点。由此,可以相邻遍历完整个二叉搜索树。

注意:pre要定义在递归函数外部,因此所求值可能在右子树中取得,如果所求值始终在某子树的根节点处才能决定,那么就后序遍历并将结果变量定义为函数参数进行回溯,如果所求值在底层出现就前序遍历回溯。

2.代码实现

class Solution {
public:int result = INT_MAX;TreeNode* pre = NULL;//前一个节点保存到外面,不能作为参数传递//否则每次更新完的值由于递归特性导致又变回原来的值。void traversal(TreeNode* cur){if(cur == NULL) return ;traversal(cur->left);//前if(pre){result = min(result,cur->val-pre->val);//中}pre = cur;//节点处理依旧是在中间节点进行的,否则右子树遍历时无法将pre更新为中间结点traversal(cur->right);//后}int getMinimumDifference(TreeNode* root) {traversal(root);return result;}
};

二、501.二叉搜索树中的众数

题目链接/文章讲解/视频讲解: https://programmercarl.com/0501.%E4%BA%8C%E5%8F%89%E6%90%9C%E7%B4%A2%E6%A0%91%E4%B8%AD%E7%9A%84%E4%BC%97%E6%95%B0.html
状态:已解决

1.思路

        这道题还是考查对二叉搜索树的掌握,跟前面的几道题差不多,第一眼还是双指针。然后cur和pre相等就将现在的计数器curNum+1,否则计数器又从1开始计;并且相等的情况下加1后还要和现在记录的最大出现值maxNum作比较,若是大于等于它就将值放入结果数组,并更新maxNum的值。这么想着,我就开始写了,写了一半发现,好像不能判断什么时候的maxNum值是最终的最大值,因此想着遍历两遍,第一遍求出maxNum的最终值,第二遍将curNum等于maxNum的值放入结果集中。

        然而,看了视频之后发现这么做果然更复杂,实际我们依然可以保持原有做法,在curNum大于maxNum相等时,将值放入结果数组,只是需要在更新maxNum值的时候,将结果数组清空,这样最后在结果集中被保留下来的就是最后的答案了。

2.代码实现

class Solution {
public:vector<int> result;int maxNum = -100005;int curNum = 0;TreeNode* pre = NULL;void traversal(TreeNode* cur){if(cur == NULL) return;traversal(cur->left);if(pre == NULL) curNum=1;else if(pre->val == cur->val){curNum ++;}else{curNum = 1;}if(curNum == maxNum){result.push_back(cur->val);}if(curNum>maxNum){maxNum = curNum;result.clear();result.push_back(cur->val);}pre=cur;traversal(cur->right);}vector<int> findMode(TreeNode* root) {traversal(root);return result;}
};

三、236. 二叉树的最近公共祖先

题目链接/文章讲解/视频讲解: https://programmercarl.com/0236.%E4%BA%8C%E5%8F%89%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
状态:已解决

1.思路

        这里所求值在上层,也就是根节点处才能决定,因此要用后序遍历的方法,最后在根节点处进行判断(即用变量left、right接住返回值,这个left、right后序还有逻辑处理的需要,也就是后序遍历中处理中间节点的逻辑(也是回溯))。思路还是简单,有几个小的问题:

(1)如果left 和 right都不为空,说明此时root就是最近公共节点。这个比较好理解

(2)如果left为空,right不为空,就返回right,说明目标节点是通过right返回的,反之依然。(代表这个子树中是出现过目标节点的)

(3)left和right都为空,则说明未出现目标节点,返回空。 

 

2.代码实现

class Solution {
public:TreeNode * result = nullptr;bool first = true;TreeNode* traversal(TreeNode* node, TreeNode* p, TreeNode* q){if(node == NULL) return NULL;if(node->val == p->val || node->val == q->val){return node;}//已经出现目标节点就要立即返回,不再向下遍历了。TreeNode* lp = traversal(node->left,p,q);//左TreeNode* rq = traversal(node->right,p,q);//右//中间处理if(lp && rq){return node;} if(lp && !rq) return lp;else if(!lp && rq) return rq;else return NULL;}TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {return traversal(root,p,q);;}
};

这篇关于代码随想录算法训练营第二十一天| LeetCode 530.二叉搜索树的最小绝对差 、501.二叉搜索树中的众数、236. 二叉树的最近公共祖先的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

哈希leetcode-1

目录 1前言 2.例题  2.1两数之和 2.2判断是否互为字符重排 2.3存在重复元素1 2.4存在重复元素2 2.5字母异位词分组 1前言 哈希表主要是适合于快速查找某个元素(O(1)) 当我们要频繁的查找某个元素,第一哈希表O(1),第二,二分O(log n) 一般可以分为语言自带的容器哈希和用数组模拟的简易哈希。 最简单的比如数组模拟字符存储,只要开26个c

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

康拓展开(hash算法中会用到)

康拓展开是一个全排列到一个自然数的双射(也就是某个全排列与某个自然数一一对应) 公式: X=a[n]*(n-1)!+a[n-1]*(n-2)!+...+a[i]*(i-1)!+...+a[1]*0! 其中,a[i]为整数,并且0<=a[i]<i,1<=i<=n。(a[i]在不同应用中的含义不同); 典型应用: 计算当前排列在所有由小到大全排列中的顺序,也就是说求当前排列是第

认识、理解、分类——acm之搜索

普通搜索方法有两种:1、广度优先搜索;2、深度优先搜索; 更多搜索方法: 3、双向广度优先搜索; 4、启发式搜索(包括A*算法等); 搜索通常会用到的知识点:状态压缩(位压缩,利用hash思想压缩)。

hdu1240、hdu1253(三维搜索题)

1、从后往前输入,(x,y,z); 2、从下往上输入,(y , z, x); 3、从左往右输入,(z,x,y); hdu1240代码如下: #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#inc

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

综合安防管理平台LntonAIServer视频监控汇聚抖动检测算法优势

LntonAIServer视频质量诊断功能中的抖动检测是一个专门针对视频稳定性进行分析的功能。抖动通常是指视频帧之间的不必要运动,这种运动可能是由于摄像机的移动、传输中的错误或编解码问题导致的。抖动检测对于确保视频内容的平滑性和观看体验至关重要。 优势 1. 提高图像质量 - 清晰度提升:减少抖动,提高图像的清晰度和细节表现力,使得监控画面更加真实可信。 - 细节增强:在低光条件下,抖

【数据结构】——原来排序算法搞懂这些就行,轻松拿捏

前言:快速排序的实现最重要的是找基准值,下面让我们来了解如何实现找基准值 基准值的注释:在快排的过程中,每一次我们要取一个元素作为枢纽值,以这个数字来将序列划分为两部分。 在此我们采用三数取中法,也就是取左端、中间、右端三个数,然后进行排序,将中间数作为枢纽值。 快速排序实现主框架: //快速排序 void QuickSort(int* arr, int left, int rig

poj1330(LCA最近公共祖先)

题意:求最近公共祖先 思路:之前学习了树链剖分,然后我就用树链剖分的一小部分知识就可以解这个题目了,记录每个结点的fa和depth。然后查找时,每次将depth大的结点往上走直到x = y。 代码如下: #include<iostream>#include<algorithm>#include<stdio.h>#include<math.h>#include<cstring>

活用c4d官方开发文档查询代码

当你问AI助手比如豆包,如何用python禁止掉xpresso标签时候,它会提示到 这时候要用到两个东西。https://developers.maxon.net/论坛搜索和开发文档 比如这里我就在官方找到正确的id描述 然后我就把参数标签换过来