Q1.翻转二叉树(easy)

如题所示

示例:

输入:

     4
/ \
2 7
/ \ / \
1 3 6 9 输出: 4
/ \
7 2
/ \ / \
9 6 3 1 来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/invert-binary-tree

这道题目起源于一个非常搞笑的事件:据说大名鼎鼎的Mac软件包管理工具Homebrew的作者,因为做不出这道在leetcode上难度为easy的题,被谷歌公司拒了。。。

谷歌:我们90%的工程师使用您编写的软件(Homebrew),但是您却无法在面试时在白板上写出翻转二叉树这道题,这太糟糕了。

如何看待 Max Howell 被 Google 拒绝?​www.zhihu.com

格式要求

/**
* Definition for a binary tree node.
* function TreeNode(val) {
* this.val = val;
* this.left = this.right = null;
* }
*/
/**
* @param {TreeNode} root
* @return {TreeNode}
*/
var invertTree = function(root) {
// 编码
};

分析:二叉树遍历

思路就是遍历二叉树的每一个节点,然后把左右链接替换一下就可以了。前序/中序/后序 都可以。如下所示

具体代码

var invertTree = function(root) {
traveral(root);
return root;
}; function traveral(node) {
if (node === null) return;
traveral(node.left);
traveral(node.right);
const temp = node.right;
node.right = node.left;
node.left = temp;
}

Q2.二叉树的右视图(middle)

给定一棵二叉树,想象自己站在它的右侧,按照从顶部到底部的顺序,返回从右侧所能看到的节点值。

输入: [1,2,3,null,5,null,4]
输出: [1, 3, 4]
解释: 1 <---
/ \
2 3 <---
\ \
5 4 <--- 输入: [1,2,3,null,5,null,null]
输出: [1, 3, 5]
解释:
1 <---
/ \
2 3 <---
\
5 <--- 来源:LeetCode
链接:https://leetcode-cn.com/problems/binary-tree-right-side-view

格式要求

/**
* Definition for a binary tree node.
* function TreeNode(val) {
* this.val = val;
* this.left = this.right = null;
* }
*/ /**
* @param {TreeNode} root
* @return {number[]}
*/
var rightSideView = function(root) {
// 编码
}

分析:层序遍历

题目的思路很明显,对二叉树进行层序遍历,然后取得每一层的最后一个节点。放到一个数组里最后返回。

1.我们可以设置一个队列存放依次遍历的节点对象。

2.使用两层循环

  • 内层循环通过不断出队列的方式遍历当前层的节点,同时通过左右链接收集下一层节点

  • 外层循环判断队列长度>0时就继续运行,从而实现逐层迭代

3.在每次内层循环中获取最右端的非空节点

具体代码

var rightSideView = function(root) {
if (!root) return [];
const queue = [];
const arrRS = [];
// 先保存根结点,也就是第一层二叉树
queue.push(root);
while (queue.length > 0) {
// 将队列长度先保存到一个变量里面
// 表示的是上一层的节点的数量
let length = queue.length;
let temp = null;
// 遍历上一层节点,将它们的子节点加入队列中,收集得到二叉树的下一层
for (let i = 0; i < length; i++) {
// 出队列,并获得返回的父节点
const node = queue.shift();
// 每次都用当前节点的val覆盖temp
// temp最后会等于当前层最右的一个非空节点的val值
if (node.val) temp = node.val;
// 收集当前节点的左节点和右节点,从而得到下一层
if (node.left) queue.push(node.left);
if (node.right) queue.push(node.right);
}
// 收集每一层的最右节点
arrRS.push(temp);
}
return arrRS;
};

Q3.二叉树中的最大路径和(difficult)

给定一个非空二叉树,返回其最大路径和。

本题中,路径被定义为一条从树中任意节点出发,达到任意节点的序列。该路径至少包含一个节点,且不一定经过根节点。

示例1:
输入: [1,2,3] 1
/ \
2 3 输出: 6 示例2:
输入: [-10,9,20,null,null,15,7] -10
/ \
9 20
/ \
15 7 输出: 42 来源:LeetCode
链接:https://leetcode-cn.com/problems/binary-tree-maximum-path-sum/

格式要求

/**
* Definition for a binary tree node.
* function TreeNode(val) {
* this.val = val;
* this.left = this.right = null;
* }
*/
/**
* @param {TreeNode} root
* @return {number}
*/
var maxPathSum = function(root) {
// 编码
};

思路分析

1.整体思路:通过后序遍历,自底向上计算。

因为后序遍历的计算过程是:左节点-右节点-根结点。 所以通过这种遍历方式,我们可以在计算两个子节点的基础上,推断当这两个节点到父节点的最大路径和。然后不断向上累加,去计算最大值。

同时在每个节点都通过Math.max更新当前的最大值,直到回归到根结点的时候,也就能比较出最大值来了。

2.路径的单一性: 当一个节点是只是作为一个中间节点,而不是一个根节点的时候:左节点和右节点只能选择一个作为经过的路径。 因为路径是“单一”的而不是“分叉”的

例如下面的图示中, 当我们通过比较选择9-7-10这条的时候,节点8就不在路径内了

3.根节点的连接性:当一个节点作为根节点的时候,它可以将两个子树的路径连接起来

4. 对于两个子节点的累加值A,B,分3种情况讨论

  • A>0,B>0: 选择Math.max(A,B)作为经过路径

  • A>0,B<0: 选择A作为经过路径

  • A<0,B>0: 选择B作为经过路径

  • A<0,B<0: A,B都不选

综上所述

我们的思路是:

  1. 后序遍历,自底向上计算

  2. 对于每个节点,假设它是根结点,计算它联合两个子树路径后的最大值

  3. 对于每个节点,假设它是中间节点,选择两条中较大的一条子树作为路径

  4. 对于2,3分上面的四种情况进行分别处理

具体代码

// 1.考虑全为负数的情况
// 2.考虑当前节点为负的情况
let max = Number.MIN_VALUE;
var maxPathSum = function(root) {
max = root.val;
traveral(root);
return max;
}; function traveral(node) {
if (node === null) return 0;
const a = traveral(node.left);
const b = traveral(node.right);
let v = node.val;
if (a >= 0 && b >= 0) {
max = Math.max(max, v + a + b);
v += Math.max(a, b);
} else if (a >= 0) {
max = Math.max(max, v + a);
v += a;
} else if (b >= 0) {
max = Math.max(max, v + b);
v += b;
}
return v;
} function TreeNode(val) {
this.val = val;
this.left = this.right = null;
}
 

本文完

JS刷算法题:二叉树的更多相关文章

  1. 19道常见的JS面试算法题

    最近秋招也做了多多少少的面试题,发现除了基础知识外,算法还是挺重要的.特意整理了一些常见的算法题,添加了自己的理解并实现. 除此之外,建议大家还可以刷刷<剑指offer>.此外,左神在牛客 ...

  2. js的算法题

    1.统计一个字符串中出现最多的字母 给出一个字符串,统计出现次数最多的字母.如:“wqeqwhixswiqhdxsq”,其中出现最多的是q. js算法的实现 function findMax(str) ...

  3. JS基础算法题(二)

    1.1 数组去重的五种方法 数组去重:将数组中重复的元素去掉 JS数组没有删除具体元素的删除(只能删掉值,删不掉元素的索引),可以使用另外一个结构来进行存储 新数组 新对象 JS数组虽然本质可以删除第 ...

  4. FCC JS基础算法题(5):Return Largest Numbers in Arrays(找出多个数组中的最大数)

    题目描述: 找出多个数组中的最大数右边大数组中包含了4个小数组,分别找到每个小数组中的最大值,然后把它们串联起来,形成一个新数组.提示:你可以用for循环来迭代数组,并通过arr[i]的方式来访问数组 ...

  5. FCC JS基础算法题(4):Title Case a Sentence(句中单词首字母大写)

    题目描述: 确保字符串的每个单词首字母都大写,其余部分小写.像'the'和'of'这样的连接符同理. 算法: function titleCase(str) { // 转小写及分割成数组 var st ...

  6. FCC JS基础算法题(2):Check for Palindromes(检查回文字符串)

    题目描述: 如果给定的字符串是回文,返回true,反之,返回false.如果一个字符串忽略标点符号.大小写和空格,正着读和反着读一模一样,那么这个字符串就是palindrome(回文).注意你需要去掉 ...

  7. FCC JS基础算法题(1):Factorialize a Number(计算一个整数的阶乘)

    题目描述: 如果用字母n来代表一个整数,阶乘代表着所有小于或等于n的整数的乘积.阶乘通常简写成 n!例如: 5! = 1 * 2 * 3 * 4 * 5 = 120. 算法: function fac ...

  8. FCC JS基础算法题(0):Reverse a String(翻转字符串)

    题目描述: 先把字符串转化成数组,再借助数组的reverse方法翻转数组顺序,最后把数组转化成字符串.你的结果必须得是一个字符串. 算法: function reverseString(str) { ...

  9. FCC JS基础算法题(10):Falsy Bouncer(过滤数组假值)

    题目描述: 删除数组中的所有假值.在JavaScript中,假值有false.null.0."".undefined 和 NaN. 使用filter方法,过滤掉生成的 Boolea ...

随机推荐

  1. 洛谷$P3645\ [APIO2015]$雅加达的摩天楼 最短路

    正解:最短路 解题报告: 传送门$QwQ$ 考虑暴力连边,发现最多有$n^2$条边.于是考虑分块 对于长度$p_i$小于等于$\sqrt(n)$的边,建立子图$d=p_i$.说下关于子图$d$的定义? ...

  2. Spring 配置内容外部化

  3. ecshop数据结构

    ecshop数据结构  2.7.2版本,数据库表 共88张表 注: 1.颜色为蓝色的字,有待讨论验证的地方. 2.颜色为红色的字,是新增的字段.(改文档是基于网上下载的老版本的数据字典修改而成,已经检 ...

  4. web前端安全——常见的web攻击方法

    面试题:你所了解的web攻击? 1.xss攻击 2.CSRF攻击 3.网络劫持攻击 4.控制台注入代码 5.钓鱼 6.DDoS攻击 7.SQL注入攻击 8.点击劫持 一.xss攻击 XSS攻击:跨站脚 ...

  5. 基于 HTML5 WebGL + WebVR 的 3D 虚实现实可视化培训系统

    前言 2019 年 VR, AR, XR, 5G, 工业互联网等名词频繁出现在我们的视野中,信息的分享与虚实的结合已经成为大势所趋,5G 是新一代信息通信技术升级的重要方向,工业互联网是制造业转型升级 ...

  6. oracle中使用pl/sql进行的文件读写操作

    第一次知道,可以使用pl/sql来进行文件的读写操作,嘿嘿,简单的试了下可行. 基本步骤如下: SQL> conn sys/sys@orcl as sysdba 已连接. SQL> cre ...

  7. 2018铁人三项测评题 IOS99

    下面这一部分是我从网上复制过来的, 2.IOS 解题链接:http://ctf4.shiyanbar.com/web/IOS/index.php 这题页面中提示系统升级到了IOS99,我们可以想到修改 ...

  8. 源码详解系列(六) ------ 全面讲解druid的使用和源码

    简介 druid是用于创建和管理连接,利用"池"的方式复用连接减少资源开销,和其他数据源一样,也具有连接数控制.连接可靠性测试.连接泄露控制.缓存语句等功能,另外,druid还扩展 ...

  9. Quartz 和 springboot schedule中的cron表达式关于星期(周几)的不同表示

    一.Quartz中cron 表达式分析: quartz 官方源码(org.quartz.CronExpression)解释: Cron expressions are comprised of 6 r ...

  10. UGUI源码之EventSystem

    今天研究下UGUI的源码,先从EventSystem入手.EventSystem是用来处理点击.键盘输入以及触摸等事件的. 1.BaseInputModule EventSystem开头声明了两个变量 ...