代码随想录算法训练营第二十天| 654.最大二叉树,617.合并二叉树,700.二叉搜索树中的搜索,98.验证二叉搜索树

 题目与题解

654.最大二叉树

题目链接:654.最大二叉树

代码随想录题解:654.最大二叉树

视频讲解:又是构造二叉树,又有很多坑!| LeetCode:654.最大二叉树_哔哩哔哩_bilibili

解题思路:

        构造最大二叉树,递归非常合适。入参是用于构造的数组,返回值是构造结束后的根结点,终止条件是当前结点为空。递归体为:首先遍历数组,找到数组中最大值及其所在的位置;然后定义两个新数组,一个包含最大值左边所有数,用于构造左子树,一个包含最大值右边所有数,用于构造右子树;然后分别将两个数组输入最大二叉树构造方法,构造对应根结点的左右子树。

class Solution {public TreeNode constructMaximumBinaryTree(int[] nums) {if (nums.length == 0) return null;int maxIndex = 0;int max = nums[0];for (int i = 1; i < nums.length; i++) {if (nums[i] > max) {maxIndex = i;max = nums[i];}}int[] leftNums = new int[maxIndex];for (int i = 0; i < maxIndex; i++) {leftNums[i] = nums[i];}int[] rightNums = new int[nums.length - maxIndex - 1];for (int i = maxIndex + 1; i < nums.length; i++) {rightNums[i - maxIndex - 1] = nums[i];}TreeNode node = new TreeNode(max, constructMaximumBinaryTree(leftNums), constructMaximumBinaryTree(rightNums));return node;}
}

看完代码随想录之后的想法 

        思路跟随想录第二个方法是一致的,第一个方法略有点复杂,还需要多加一次判断和取值,先pass。

        写法上我还有待改进,因为不想写多余的递归函数,所以就直接使用该函数为递归体了,但是也造成每次调用的时候,都要new两个新的数组,效率会降低(Java不像C++可以直接传地址)。像答案里面调用的递归函数入参写成原数组和头尾index就比较方便。此外,只有一个元素时,可以提前判断返回,这样也不需要额外再求一次最大值和构建左右子树的数组了。

class Solution {public TreeNode constructMaximumBinaryTree(int[] nums) {return constructMaximumBinaryTree1(nums, 0, nums.length);}public TreeNode constructMaximumBinaryTree1(int[] nums, int leftIndex, int rightIndex) {if (rightIndex - leftIndex < 1) {// 没有元素了return null;}if (rightIndex - leftIndex == 1) {// 只有一个元素return new TreeNode(nums[leftIndex]);}int maxIndex = leftIndex;// 最大值所在位置int maxVal = nums[maxIndex];// 最大值for (int i = leftIndex + 1; i < rightIndex; i++) {if (nums[i] > maxVal){maxVal = nums[i];maxIndex = i;}}TreeNode root = new TreeNode(maxVal);// 根据maxIndex划分左右子树root.left = constructMaximumBinaryTree1(nums, leftIndex, maxIndex);root.right = constructMaximumBinaryTree1(nums, maxIndex + 1, rightIndex);return root;}
}

遇到的困难

        写起来不难,细节优化有待提升。

617.合并二叉树

题目链接:617.合并二叉树

代码随想录题解:617.合并二叉树

视频讲解:一起操作两个二叉树?有点懵!| LeetCode:617.合并二叉树_哔哩哔哩_bilibili

解题思路:

        操作两个树和操作一个树的方法其实差不多,递归每次同时传入位置相同的结点就行。因此递归的入参为两个树相同位置的子树根结点,返回值为合并后的根节点,终止条件是两个结点都为空。递归体为,如果两个结点都为空,不做操作直接返回null;如果其中一个结点为空,直接返回非空结点;如果都不为空,则new一个新的结点,其值为两个结点值之和,左子树和右子树都分别调用当前函数,入参分别为两个结点的左子树和右子树,最后返回该新结点。

class Solution {public TreeNode mergeTrees(TreeNode root1, TreeNode root2) {if (root1 == null && root2 == null) return null;else if (root1 != null && root2 == null) {return root1;} else if (root1 == null && root2 != null) {return root2;} else {return new TreeNode(root1.val + root2.val,mergeTrees(root1.left, root2.left),mergeTrees(root1.right, root2.right));}}
}

看完代码随想录之后的想法 

        递归法思路一致,迭代法也比较好理解,有符合条件的左右子树才加入队列,否则直接赋值,避免了队列中的元素有重复或者缺漏。不过写起来有些麻烦,就看看吧。

class Solution {
public:TreeNode* mergeTrees(TreeNode* t1, TreeNode* t2) {if (t1 == NULL) return t2;if (t2 == NULL) return t1;queue<TreeNode*> que;que.push(t1);que.push(t2);while(!que.empty()) {TreeNode* node1 = que.front(); que.pop();TreeNode* node2 = que.front(); que.pop();// 此时两个节点一定不为空,val相加node1->val += node2->val;// 如果两棵树左节点都不为空,加入队列if (node1->left != NULL && node2->left != NULL) {que.push(node1->left);que.push(node2->left);}// 如果两棵树右节点都不为空,加入队列if (node1->right != NULL && node2->right != NULL) {que.push(node1->right);que.push(node2->right);}// 当t1的左节点 为空 t2左节点不为空,就赋值过去if (node1->left == NULL && node2->left != NULL) {node1->left = node2->left;}// 当t1的右节点 为空 t2右节点不为空,就赋值过去if (node1->right == NULL && node2->right != NULL) {node1->right = node2->right;}}return t1;}
};

遇到的困难

        无

700.二叉搜索树中的搜索

题目链接:​​​​​​​700.二叉搜索树中的搜索

代码随想录题解:700.二叉搜索树中的搜索

视频讲解:不愧是搜索树,这次搜索有方向了!| LeetCode:700.二叉搜索树中的搜索_哔哩哔哩_bilibili

解题思路:

        二叉搜索树本身就是通过一定的排序构造的,所以搜索效率极高。一般二叉搜索树的左子树所有结点比根结点小,右子树的所有结点比根结点大,所以搜索的时候只要通过递归或者迭代的方式,每层比较一次,就可以快速找到结果了。

递归法:入参为当前根结点和待搜索的值,返回值是布尔值,表示搜到与否,终止条件是搜到结果或当前结点为空(即不存在符合条件的值)。递归体为,如果当前结点的值等于目标值,直接返回true,如果当前结点值小于目标值,继续搜索该结点右子树,否则搜索该结点左子树。

class Solution {public TreeNode searchBST(TreeNode root, int val) {// 递归if (root == null || root.val == val) return root;if (val < root.val) return searchBST(root.left, val);else return searchBST(root.right, val);}
}

迭代法:按层序遍历的思路,用队列保存符合要求的结点,如果队首结点的值等于目标值,直接返回true,如果队首结点值小于目标值,将其右结点加入队列,否则将其左结点加入队列,直到队列为空返回false。

class Solution {public TreeNode searchBST(TreeNode root, int val) {// 迭代队列版if (root == null || root.val == val) return root;Queue<TreeNode> q = new LinkedList<>();q.add(root);while (!q.isEmpty()) {TreeNode node = q.poll();if (val == node.val) return node;if (val < node.val && node.left != null) q.add(node.left);if (val > node.val && node.right != null) q.add(node.right);}return null;}
}

看完代码随想录之后的想法 

        递归法是一样的,但是迭代法写复杂了,事实上都不需要队列或者栈去存数字,只要像链表那样一层一层直接往下搜索就好,非常简单。

class Solution {public TreeNode searchBST(TreeNode root, int val) {// 迭代极简版if (root == null || root.val == val) return root;if (val < root.val) root = root.left;else root = root.right;return searchBST(root, val);}
}

遇到的困难

        基础题,不难写

98.验证二叉搜索树

题目链接:98.验证二叉搜索树

代码随想录题解:98.验证二叉搜索树

视频讲解:你对二叉搜索树了解的还不够! | LeetCode:98.验证二叉搜索树_哔哩哔哩_bilibili

解题思路:

        一开始想用递归完成,利用二叉搜索树的性质,比较每个树根节点的值和其左右结点的值,如果左结点小于根结点,右结点大于根结点,该节点就符合要求,继续递归的判断左右子树是否符合要求即可。

        但是这样做会出错,原因是二叉搜索树要求的是左子树的每一个结点都小于根结点,右子树的每一个结点都大于根结点,只满足一层是不够的。然后看了一下leetcode上给的题解,用了一个新的递归函数,入参是当前节点和历史最大值及最小值,返回值是布尔值,表示树是否是二叉搜索树,终止条件为当前结点为空或者其值比历史最小值小或比历史最大值大。递归体为将当前节点的左右子树根结点分别放入递归函数,放左子树时,其输入的历史最大值为当前根结点的值,放右子树时,其输入的历史最小值为当前根结点的值。        

class Solution {TreeNode pre;public boolean isValidBST(TreeNode root) {if (root == null) return true;return isValidBST(root, Long.MIN_VALUE, Long.MAX_VALUE);}public boolean isValidBST(TreeNode root, long min, long max) {if (root == null) return true;if (root.val <= min || root.val >= max) return false;return isValidBST(root.left, min, root.val) && isValidBST(root.right, root.val, max);}
}

看完代码随想录之后的想法 

        代码随想录给出的前序遍历得到数组判断该数组是否是递增数组的思路很清晰,虽然多了空间复杂度,但是很好理解。

// 简洁实现·中序遍历
class Solution {private long prev = Long.MIN_VALUE;public boolean isValidBST(TreeNode root) {if (root == null) {return true;}if (!isValidBST(root.left)) {return false;}if (root.val <= prev) { // 不满足二叉搜索树条件return false;}prev = root.val;return isValidBST(root.right);}
}

        另一种递归只记录了用最大值去判断是否符合要求,着实是有点看不懂了,还是跳过吧。

遇到的困难

        第一次写完上传的时候就调入了陷阱,二叉搜索树的定义没有理解透彻,碰到不符合要求的树就出错了。参考答案写出含上下限的递归函数后,又因为测试用例里面包含了Integer的最大值和最小值,与初始化的值相同了,所以再次出错,仔细看了答案以后发现用的是long而非int变量,这样就不会有边界问题。也是蛮tricky的。

今日收获

        再次加深了一下二叉树的印象,学习二叉搜索树的定义和用法。递归函数的选择真的很奇妙,好好学习。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/770094.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

数据库及中表的创建和管理

目录 创建数据库 使用数据库(使用,查看信息) 修改数据库(删除,修改)

promethus的安装使用

1、# 软件下载地址 https://prometheus.io/download/ https://grafana.com/grafana/download https://prometheus.io/download/ Prometheus是一套开源的监控&报警&时间序列数据库的组合,起始是由SoundCloud公司开发的。 Prometheus 的优点 1、非常少的外部依赖,安装…

OceanBase4.2.2.1 单机集群在ArmX86安装(自测记录)

OceanBase OceanBase就不必多加介绍了&#xff0c;本次主要是分享对于它的安装使用&#xff0c;先说说背景&#xff0c;首先接触是因为信创国产化的要求&#xff0c;为满足支持国产化&#xff0c;安装了Arm架构下版本4.0.0&#xff0c;满足支持通过。后来项目实际使用&#xff…

由浅到深认识Java语言(20):包装类

该文章Github地址&#xff1a;https://github.com/AntonyCheng/java-notes 在此介绍一下作者开源的SpringBoot项目初始化模板&#xff08;Github仓库地址&#xff1a;https://github.com/AntonyCheng/spring-boot-init-template & CSDN文章地址&#xff1a;https://blog.c…

JavaScript中实现数组去重

在JavaScript中实现数组去重是一个常见的问题&#xff0c;下面是一个使用不同方法实现数组去重的示例代码&#xff1a; 方法一&#xff1a;使用Set对象 javascript function uniqueArray(arr) { return [...new Set(arr)]; } const arr [1, 2, 3, 3, 4, 5, 5, 6]; cons…

leetcode刷题日记-外观数组

题目描述 解题思路 初始化字符串 init 为 “1”&#xff0c;作为外观数列的第一项。 通过循环迭代生成外观数列的下一项&#xff0c;循环次数为 n-1&#xff0c;因为已经初始化了第一项。 在每次迭代中&#xff0c;通过两个指针 pos 和 start 来遍历当前项 init&#xff0c;po…

C++中的枚举类型

C中的enum&#xff08;枚举&#xff09;类型是一种用户定义的类型&#xff0c;用于表示一组整数值&#xff0c;每个值都有对应的名称&#xff0c;增强了代码的可读性和可维护性。 1. 基本枚举类型 基本的枚举类型定义了一组命名的整数常量。 enum Color {RED,GREEN,BLUE };C…

创业之路:探索如何在Facebook上创业成功

引言 在当今数字化时代&#xff0c;社交媒体已成为创业者们开展业务、建立品牌和推广产品的重要平台之一。作为全球最大的社交媒体平台之一&#xff0c;Facebook为创业者提供了丰富的机会和资源。本文将探讨如何在Facebook上创业成功的关键因素和实践方法&#xff0c;帮助创业…

Spring Cloud: openFegin

文章目录 一、什么是openFegin 一、什么是openFegin Spring Cloud OpenFeign 是一个声明式的 Web Service 客户端&#xff0c;它使得编写 Web Service 客户端变得更加简单。OpenFeign 整合了 Ribbon 和 Hystrix&#xff0c;提供了负载均衡和容错机制。同时&#xff0c;OpenFei…

语 句 篇

文章目录 1. if 语句2. for 循环3. while 循环 1. if 语句 if 语句用于基于某个条件执行代码。如果条件为真&#xff08;True&#xff09;&#xff0c;则执行相应的代码块&#xff1b;如果条件为假&#xff08;False&#xff09;&#xff0c;则跳过该代码块。 基本语法&#…

第九届蓝桥杯大赛个人赛省赛(软件类)真题C 语言 A 组-第几个幸运数字

幸运数字是可以被3,5,7任一整除的数字&#xff0c;列举小明号码内的所有可能组合并计数。注意别忘了把1占的一位减去。 #include<stdio.h> typedef long long ll; int main(){long long ans 0, n 59084709587505LL;for(ll i 1; i < n; i * 3){//计算小于等于n的数…

关于MySQL查询JSON的语法糖

1. 根据单JSON对象查询 select count(1) from report_configuration r where json_extract(r.report_configuration,$.dataSetName) :dataSetName 2.纯数组JSON查询 one代表就遇到第一个就返回 固定写法 SELECT count(1) FROM statistics_property where data_set_name …

AI程序员的诞生会对程序员有多大影响?

近期&#xff0c;全球首位AI程序员Devin的出场&#xff0c;不禁让我想到了一个有趣的问题&#xff1a;AI程序员会不会抢程序员的饭碗呢&#xff1f;先别着急下结论&#xff01;虽然AI技术在编程领域越来越广泛&#xff0c;但它真的能完全替代我们程序员吗&#xff1f; 目前的AI…

不愧是淘天,全方位八股拷打

恭喜发现宝藏&#xff01;搜索公众号【TechGuide】回复公司名&#xff0c;解锁更多新鲜好文和互联网大厂的笔经面经&#xff0c;目前已更新至美团、微软… 作者TechGuide【全网同名】 基本情况 投递岗位&#xff1a;后台开发 投递部门&#xff1a;阿里淘天 招聘类型&#xf…

数据结构中排序算法

介绍 排序算法是计算机科学中的一类算法&#xff0c;用于对元素序列进行排序&#xff0c;以便按照某种特定的顺序&#xff08;如升序或降序&#xff09;组织数据。这些算法在软件开发和数据处理中扮演着至关重要的角色&#xff0c;因为它们可以提高搜索效率、优化数据结构的访…

计算机程序的编译和链接

c语言中的小小白-CSDN博客c语言中的小小白关注算法,c,c语言,贪心算法,链表,mysql,动态规划,后端,线性回归,数据结构,排序算法领域.https://blog.csdn.net/bhbcdxb123?spm1001.2014.3001.5343 给大家分享一句我很喜欢我话&#xff1a; 知不足而奋进&#xff0c;望远山而前行&am…

下载网页上的在线视频 网络视频 视频插件下载

只需要在浏览器上安装一个插件&#xff0c;就可以下载大部分的视频文件&#xff0c;几秒到一两个小时的视频&#xff0c;基本都不是问题。详细解决如下&#xff1a; 0、因为工作需要&#xff0c;需要获取某网站上的宣传视频&#xff0c;我像往常一样&#xff0c;查看视频的url…

Schemdraw小白从入门到放弃---原理工具书

文章目录 序版本最简单的例子一、总体思路二、元件2.1 color习题 2.2 label2.3 length 三、元件的连接3.1 延续性习题 3.2 方向习题 3.3 接线点习题3.3.1 默认激活anchor与沉默anchor3.3.2 切换鼠标焦点机制3.3.2.1 at函数规定元件的start接在哪个anchor上3.3.2.2 to函数规定元…

重磅!一起做个淘宝的简易布局!(超详细)

你好&#xff0c;我是云桃桃。 一个希望帮助更多朋友快速入门 WEB 前端程序媛。 因为之前的学习内容&#xff0c;今天&#xff0c;我们可以来综合运用一下标签和 CSS 样式&#xff0c;做一个简易的淘宝网页大体布局了&#xff0c;如图。 咱们今天要做成这样子&#xff01; 里面…

4.1 RK3399项目开发实录-案例开发之MIPI 摄像头开发(wulianjishu666)

嵌入式从零到项目开发全套例程资料 链接&#xff1a;https://pan.baidu.com/s/1ksCQN__jD8ZrJhw8sWzhwQ?pwdvvfz 3.2. MIPI 摄像头 带有 MIPI CSI 接口的 RK3399 板子都添加了双 MIPI 摄像头 OV13850 的支持&#xff0c;应用中也添加了摄像头的例子。下面介绍一下相关配置。…