Day21|二叉树part07:530.二叉搜索树的最小绝对差、501.二叉搜索树中的众数、236. 二叉树的最近公共祖先

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

530. *二叉搜索树的最小绝对差(双指针题型)

众所周知二叉搜索树的中序遍历序列是一个有序数组,因此最基本的方法就是遍历得到中序序列再进行计算,实际上可以用双指针法,记录中序遍历前一个指针和当前指针的差值:

class Solution {private int res = Integer.MAX_VALUE;private TreeNode pre = null;private void traversal(TreeNode cur){if(cur == null){return;}traversal(cur.left);if(pre != null){res = Math.min(res, Math.abs(cur.val - pre.val));}pre = cur;traversal(cur.right);}public int getMinimumDifference(TreeNode root) {traversal(root);return res;}
}

501.二叉搜索树中的众数

这里分两种方法,一种是当作普通的二叉树来做,一种是利用BST的性质来做。

  • 使用普通的二叉树,遍历一遍,将频率用Map存储,然后找到频率最高的。
class Solution {private Map<Integer, Integer> times = new HashMap<>();void traversal(TreeNode node) {if (node == null) {return;}traversal(node.left);times.put(node.val, times.getOrDefault(node.val, 0) + 1);traversal(node.right);}public int[] findMode(TreeNode root) {//1. 得到(值,频率)的maptraversal(root);int maxCount = 0;for(Integer count: times.values()){maxCount = Math.max(maxCount, count);}//2. 将频率最高的key找到List<Integer> res = new ArrayList<>();for (Map.Entry<Integer, Integer> entry : times.entrySet()) {if (entry.getValue() == maxCount) {res.add(entry.getKey());}}//3. 根据key去找value并转化成数组int[] result = new int[res.size()];for (int i = 0; i < res.size(); i++) {result[i] = res.get(i);}return result;}}
  • java的数组真的太麻烦了,又是Map转List再转数组的,不要乱;

利用BST性质的解法:

class Solution {public void test(){TreeNode root = new TreeNode(0);
//            root.right = new TreeNode(2);
//            root.right.left = new TreeNode(2);findMode(root);}private List<Integer> res = new ArrayList<>();private int count = 0;private int maxCount = 0;private TreeNode pre = null;void traversal(TreeNode node) {if(node == null){return;}traversal(node.left);if(pre == null){count = 1;}else if(node.val == pre.val){count++;}else{count = 1;}pre = node;if(count == maxCount){res.add(node.val);}if(count > maxCount){maxCount = count;res.clear();res.add(node.val);}traversal(node.right);}public int[] findMode(TreeNode root) {traversal(root);int[] array = res.stream().mapToInt(i->i).toArray();return array;}}
  • 也是需要两个指针一个pre一个cur,这样只遍历一次二叉树就可以获取众数节点。

236. ***二叉树的最近公共祖先(LCA)

  • 公共祖先的定义:

“对于有根树 T 的两个节点 p、q,最近公共祖先表示为一个节点 x,满足 x 是 p、q 的祖先且 x 的深度尽可能大(一个节点也可以是它自己的祖先)。”

这里是求最近公共祖先。因为root是所有节点的公共祖先 )

  • 需要从底向上遍历,那么二叉树,只能通过后序遍历(即:回溯)实现从底向上的遍历方式。
  • 解法思路:如果一个节点能够在它的左右子树中分别找到 pq,则该节点为 LCA 节点。这样可以借助find函数得到本题的解法:
  • 判断逻辑是 如果递归遍历遇到q,就将q返回,遇到p 就将p返回,那么如果 左右子树的返回值都不为空,说明此时的中节点,一定是q 和p 的最近祖先。
class Solution {private TreeNode find(TreeNode node, TreeNode p, TreeNode q){if(node == null || node == p || node == q){return node;}TreeNode left = find(node.left, p ,q);TreeNode right = find(node.right, p , q);if(left != null && right != null){return node;}else if(left != null && right == null){return left;}else if(left == null && right != null){return right;}else{return null;}}public TreeNode lowestCommonAncestor(TreeNode root, TreeNode p, TreeNode q) {return find(root, p ,q);}
}
  • 在递归函数有返回值的情况下:如果要搜索一条边,递归函数返回值不为空的时候,立刻返回,如果搜索整个树,直接用一个变量left、right接住返回值,这个left、right后序还有逻辑处理的需要,也就是后序遍历中处理中间节点的逻辑(也是回溯)

image

这篇关于Day21|二叉树part07:530.二叉搜索树的最小绝对差、501.二叉搜索树中的众数、236. 二叉树的最近公共祖先的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

C# ComboBox下拉框实现搜索方式

《C#ComboBox下拉框实现搜索方式》文章介绍了如何在加载窗口时实现一个功能,并在ComboBox下拉框中添加键盘事件以实现搜索功能,由于数据不方便公开,作者表示理解并希望得到大家的指教... 目录C# ComboBox下拉框实现搜索步骤一步骤二步骤三总结C# ComboBox下拉框实现搜索步骤一这

认识、理解、分类——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

poj1330(LCA最近公共祖先)

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

poj 1258 Agri-Net(最小生成树模板代码)

感觉用这题来当模板更适合。 题意就是给你邻接矩阵求最小生成树啦。~ prim代码:效率很高。172k...0ms。 #include<stdio.h>#include<algorithm>using namespace std;const int MaxN = 101;const int INF = 0x3f3f3f3f;int g[MaxN][MaxN];int n

poj 1287 Networking(prim or kruscal最小生成树)

题意给你点与点间距离,求最小生成树。 注意点是,两点之间可能有不同的路,输入的时候选择最小的,和之前有道最短路WA的题目类似。 prim代码: #include<stdio.h>const int MaxN = 51;const int INF = 0x3f3f3f3f;int g[MaxN][MaxN];int P;int prim(){bool vis[MaxN];

poj 2349 Arctic Network uva 10369(prim or kruscal最小生成树)

题目很麻烦,因为不熟悉最小生成树的算法调试了好久。 感觉网上的题目解释都没说得很清楚,不适合新手。自己写一个。 题意:给你点的坐标,然后两点间可以有两种方式来通信:第一种是卫星通信,第二种是无线电通信。 卫星通信:任何两个有卫星频道的点间都可以直接建立连接,与点间的距离无关; 无线电通信:两个点之间的距离不能超过D,无线电收发器的功率越大,D越大,越昂贵。 计算无线电收发器D

poj 1734 (floyd求最小环并打印路径)

题意: 求图中的一个最小环,并打印路径。 解析: ans 保存最小环长度。 一直wa,最后终于找到原因,inf开太大爆掉了。。。 虽然0x3f3f3f3f用memset好用,但是还是有局限性。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#incl

hdu 1102 uva 10397(最小生成树prim)

hdu 1102: 题意: 给一个邻接矩阵,给一些村庄间已经修的路,问最小生成树。 解析: 把已经修的路的权值改为0,套个prim()。 注意prim 最外层循坏为n-1。 代码: #include <iostream>#include <cstdio>#include <cstdlib>#include <algorithm>#include <cstri

poj 2175 最小费用最大流TLE

题意: 一条街上有n个大楼,坐标为xi,yi,bi个人在里面工作。 然后防空洞的坐标为pj,qj,可以容纳cj个人。 从大楼i中的人到防空洞j去避难所需的时间为 abs(xi - pi) + (yi - qi) + 1。 现在设计了一个避难计划,指定从大楼i到防空洞j避难的人数 eij。 判断如果按照原计划进行,所有人避难所用的时间总和是不是最小的。 若是,输出“OPETIMAL",若