【一天一大 lee】 二叉搜索树中的众数 (难度:简单)-Day20200924
时间:2022-07-26
本文章向大家介绍【一天一大 lee】 二叉搜索树中的众数 (难度:简单)-Day20200924,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。
题目:[1]
给定一个有相同值的二叉搜索树(BST),找出 BST 中的所有众数(出现频率最高的元素)。
假定 BST 有如下定义:
- 结点左子树中所含结点的值小于等于当前结点的值
- 结点右子树中所含结点的值大于等于当前结点的值
- 左子树和右子树都是二叉搜索树
例如:给定 BST [1,null,2,2],
示例:
1
2
/
2
返回[2].
提示: 如果众数超过 1 个,不需考虑输出顺序
进阶: 你可以不使用额外的空间吗?(假设由递归产生的隐式调用栈的开销不被计算在内)
抛砖引玉
思路
先不考虑进阶中不使用额外空间的逻辑:
- 声明一个 map 记录每个节点出现的次数,并且记录最大出现次数
- 遍历二叉树(DFS 或者 BFS)
- 遍历 map 找出频率等于最大出现次数的数
/**
* Definition for a binary tree node.
* function TreeNode(val) {
* this.val = val;
* this.left = this.right = null;
* }
*/
/**
* @param {TreeNode} root
* @return {number[]}
*/
var findMode = function(root) {
let map = new Map(),
maxNum = 0,
_result = []
function dfs(node) {
if (node === null) return
if (map.has(node.val)) {
map.set(node.val, map.get(node.val) + 1)
} else {
map.set(node.val, 1)
}
maxNum = Math.max(maxNum, map.get(node.val))
dfs(node.left)
dfs(node.right)
}
// 遍历二叉树记录每个数字出现次数
dfs(root)
for (let [key, value] of map) {
if (maxNum === value) _result.push(key)
}
return _result
}
加上进阶的限制的不使用额外的空间:
因为题目指定了二叉树的类型是二叉搜索树,二叉搜索树的特性就是中序遍历元素是递增的。
那么中序遍历传入的二叉树,重复出现的元素一定是相邻出现的,那么记录:
- 当前出现最高频率:maxCount
- 后面遇到的新节点:nextNode
- 后面遇到的新节点出现的频率:nextNum
当 nextNum 大于 maxCount 时则 nextNode 替换结果数组中的元素
当 nextNum 等于 maxCount 时则 nextNode 追加到结果数组中
中序遍历
var findMode = function(root) {
let maxNum = 0,
nextNode = null,
nextNum = 0,
_result = []
function dfs(node) {
if (node === null) return
dfs(node.left)
update(node.val)
dfs(node.right)
}
// 遍历二叉树记录每个数字出现次数
dfs(root)
function update(val) {
if (val === nextNode) {
++nextNum
} else {
nextNum = 1
nextNode = val
}
if (nextNum === maxNum) {
_result.push(nextNode)
}
if (nextNum > maxNum) {
maxNum = nextNum
_result = [nextNode]
}
}
return _result
}
Morris 中序遍历
Morris 中序遍历:
- 如果当前节点的左子节点为空,处理当前节点,并遍历当前节点的右子节点;
- 如果当前节点的右子节点不为空,找到当前节点左子树的最右节点(该节点为当前节点中序遍历的前驱节点);
- 如果最右节点的右指针为空,将最右节点的右指针指向当前节点,遍历当前节点的左子节点;
- 如果最右节点的右指针不为空,将最右节点的右指针重新置为空(恢复树的原状),处理当前节点,并将当前节点置为其右节点;重复步骤 1 和步骤 2,直到遍历结束。
var findMode = function(root) {
let maxNum = 0,
nextNode = null,
nextNum = 0,
_result = []
let node = root,
prevNode = null
while (node !== null) {
if (node.left === null) {
update(node.val)
node = node.right
continue
}
prevNode = node.left
while (prevNode.right !== null && prevNode.right !== node) {
prevNode = prevNode.right
}
if (prevNode.right === null) {
prevNode.right = node
node = node.left
} else {
prevNode.right = null
update(node.val)
node = node.right
}
}
function update(val) {
if (val === nextNode) {
++nextNum
} else {
nextNum = 1
nextNode = val
}
if (nextNum === maxNum) {
_result.push(nextNode)
}
if (nextNum > maxNum) {
maxNum = nextNum
_result = [nextNode]
}
}
return _result
}
- 【干货】不止准确率:为分类任务选择正确的机器学习度量指标(附代码实现)
- python爬虫beautifulsoup4系列1
- 区块链入门教程
- python爬虫beautifulsoup4系列2
- python爬虫beautifulsoup4系列3
- 多元回归模型
- C++ STL之min_element()与max_element()(取容器中的最大最小值)
- RESTful API 设计最佳实践
- python爬虫beautifulsoup4系列4-子节点
- 元胞自动机实现多数分类算法
- 51Nod 1289 大鱼吃小鱼(模拟,经典好题)
- 用Metaclass实现一个精简的ORM框架
- HDU 2504 又见GCD(最大公约数与最小公倍数变形题)
- Selenium2+python自动化63-二次封装(click/send_kesy)
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- jvm线上内存问题排查
- (18)Bash输入输出重定向
- RPC 和 REST还有RESTFul到底是个什么玩意?
- 线程和线程池的几个状态值
- 阿里代码规约为什么不让使用Executors包装好线程池呢?
- Groovy、热部署和热加载(自定义类加载器)及spring loaded 部分源码分析
- Centos在线迁移到腾讯云cvm
- ClickHouse源码导读:网络IO
- 手把手教你:将ClickHouse集群迁至云上
- 直播中台iLiveSDK终端框架演变之路
- ClickHouse 数据导入实战:Kafka 篇
- pipeAsyncFunctions
- 【python-leetcode287-循环排序】寻找重复的数
- 聊聊claudb的list command
- 【python-leetcode448-循环排序】找到所有数组中消失的数字