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

相关文章

公共筛选组件(二次封装antd)支持代码提示

如果项目是基于antd组件库为基础搭建,可使用此公共筛选组件 使用到的库 npm i antdnpm i lodash-esnpm i @types/lodash-es -D /components/CommonSearch index.tsx import React from 'react';import { Button, Card, Form } from 'antd'

LeetCode--155 最小栈

题目 设计一个支持 push,pop,top 操作,并能在常数时间内检索到最小元素的栈。push(x) -- 将元素 x 推入栈中。pop() -- 删除栈顶的元素。top() -- 获取栈顶元素。getMin() -- 检索栈中的最小元素。 示例 MinStack minStack = new MinStack();minStack.push(-2);minStack.push

剑指offer(C++)--平衡二叉树

题目 输入一棵二叉树,判断该二叉树是否是平衡二叉树。 class Solution {public:bool IsBalanced_Solution(TreeNode* pRoot) {if(pRoot==NULL)return true;int left_depth = getdepth(pRoot->left);int right_depth = getdepth(pRoot->rig

剑指offer(C++)--两个链表的第一个公共结点

题目 输入两个链表,找出它们的第一个公共结点。 解法一 两个链表一定有交点的话,方法是指向短链表指针先走完,然后指向长链表,指向长链表指针后走完,指向短链表。所以,第二次走过,一定会在交点相遇。 class Solution {public:ListNode* FindFirstCommonNode( ListNode *pHead1, ListNode *pHead2) {ListN

二叉树三种遍历方式及其实现

一、基本概念 每个结点最多有两棵子树,左子树和右子树,次序不可以颠倒。 性质: 1、非空二叉树的第n层上至多有2^(n-1)个元素。 2、深度为h的二叉树至多有2^h-1个结点。 3、对任何一棵二叉树T,如果其终端结点数(即叶子结点数)为n0,度为2的结点数为n2,则n0 = n2 + 1。 满二叉树:所有终端都在同一层次,且非终端结点的度数为2。 在满二叉树中若其深度为h,则其所包含

【文末附gpt升级秘笈】腾讯元宝AI搜索解析能力升级:千万字超长文处理的新里程碑

腾讯元宝AI搜索解析能力升级:千万字超长文处理的新里程碑 一、引言 随着人工智能技术的飞速发展,自然语言处理(NLP)和机器学习(ML)在各行各业的应用日益广泛。其中,AI搜索解析能力作为信息检索和知识抽取的核心技术,受到了广泛的关注和研究。腾讯作为互联网行业的领军企业,其在AI领域的探索和创新一直走在前列。近日,腾讯旗下的AI大模型应用——腾讯元宝,迎来了1.1.7版本的升级,新版本在AI搜

LeetCode 算法:二叉树的中序遍历 c++

原题链接🔗:二叉树的中序遍历 难度:简单⭐️ 题目 给定一个二叉树的根节点 root ,返回 它的 中序 遍历 。 示例 1: 输入:root = [1,null,2,3] 输出:[1,3,2] 示例 2: 输入:root = [] 输出:[] 示例 3: 输入:root = [1] 输出:[1] 提示: 树中节点数目在范围 [0, 100] 内 -100 <= Node.

C++ 重建二叉树(递归方法)

/*** struct TreeNode {* int val;* struct TreeNode *left;* struct TreeNode *right;* TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}* };*/#include <vector>class Solution {public:/*** 代码

数据结构:二叉树详解 c++信息学奥赛基础知识讲解

目录 一、二叉树的定义 二、二叉树的形态 三、二叉树的性质 四、二叉树的存储 五、二叉树的创建与遍历(递归) 六、二叉树实现 创建二叉树 展示二叉树 1、计算数的高度 2、计算数的叶子数量 3、计算数的宽度 4、层次遍历 5、前序遍历 递归写法 非递归写法 6、中序遍历 递归写法 非递归写法 7、后序遍历 递归写法 非递归写法 8、输出根节点到所有叶