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


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

    文章链接:代码随想录最大二叉树        代码随想录合并二叉树        代码随想录二叉搜索树中的搜索        代码随想录验证二叉搜索树

    视频链接:代码随想录最大二叉树        代码随想录合并二叉树        代码随想录二叉搜索树中的搜索        代码随想录验证二叉搜索树

    1. LeetCode 654. 最大二叉树

    1.1 思路

    1. 对于这题我们要构造二叉树,凡是构造二叉树类的题目都要用前序遍历,“中左右”,先构造根节点,然后再是左子树右子树,左子树和右子树也是先“中左右”。
    2. 递归函数的参数和返回值:返回值就是这个二叉树的根节点,参数就是数组
    3. 终止条件:如果数组大小等于1说明到叶子节点了,就return new TreeNode(nums[0])。
    4. 单层递归的逻辑:找到数组的最大值及其下标,定义一个maxValue=0,index=0。遍历数组找到它们。找到最大值就定义新节点然后把数值放入,然后就是构造节点的左右子树
    5. 构造子树就要分割数组,因为终止条件是至少要有一个元素的。所以我们要判断左子树是否至少有一个元素(如果左子树没元素,说明就没有左子树咯),通过(index>0)来判断左子树是否至少有一个元素,有就切割数组,新的左子树数组就是[0,index),左闭右开,包含左端点但不包含右端点,右端点是index的,是根节点不是左子树的。然后就是 node.left=函数(新的左子树数组)
    6. 右子树同理,至少有一个元素,理由同上,通过(index
    7. 最后return node就是二叉树的根节点

    1.2 代码

    1. class Solution {
    2. public TreeNode constructMaximumBinaryTree(int[] nums) {
    3. return constructMaximumBinaryTree1(nums, 0, nums.length);
    4. }
    5. public TreeNode constructMaximumBinaryTree1(int[] nums, int leftIndex, int rightIndex) {
    6. if (rightIndex - leftIndex < 1) {// 没有元素了
    7. return null;
    8. }
    9. if (rightIndex - leftIndex == 1) {// 只有一个元素
    10. return new TreeNode(nums[leftIndex]);
    11. }
    12. int maxIndex = leftIndex;// 最大值所在位置
    13. int maxVal = nums[maxIndex];// 最大值
    14. for (int i = leftIndex + 1; i < rightIndex; i++) {
    15. if (nums[i] > maxVal){
    16. maxVal = nums[i];
    17. maxIndex = i;
    18. }
    19. }
    20. TreeNode root = new TreeNode(maxVal);
    21. // 根据maxIndex划分左右子树
    22. root.left = constructMaximumBinaryTree1(nums, leftIndex, maxIndex);
    23. root.right = constructMaximumBinaryTree1(nums, maxIndex + 1, rightIndex);
    24. return root;
    25. }
    26. }

    2. LeetCode 617. 合并二叉树

    2.1 思路

    1. 这题考察同时操作两个二叉树的能力,这题递归的情况还是前序更方便些,也更直观
    2. 递归函数的参数和返回值:返回值就是合并后的二叉树的根节点,参数是一个是二叉树t1,一个是二叉树t2
    3. 终止条件:如果t1遍历到空就返回t2对应位置的节点,如果t2遍历到空就返回t1对应位置的节点
    4. 单层递归的逻辑:我们直接改t1的结构,就不创建新树了。t1.val+=t2.val。然后是递归t1.left=函数(t1.left, t2.left);t1.right=函数(t1.right, t2.right);最后return t1就是新的树了
    5. 如果要创建新树的话就把新树的节点的值为两树之和即可,终止条件是一样的

    2.2 代码

    1. class Solution {
    2. // 递归
    3. public TreeNode mergeTrees(TreeNode root1, TreeNode root2) {
    4. if (root1 == null) return root2;
    5. if (root2 == null) return root1;
    6. root1.val += root2.val;
    7. root1.left = mergeTrees(root1.left,root2.left);
    8. root1.right = mergeTrees(root1.right,root2.right);
    9. return root1;
    10. }
    11. }

    3. LeetCode 700. 二叉搜索树中的搜索

    3.1 递归法思路

    1. 二叉搜索树中自带顺序,因此不强调前中后序。
    2. 确定递归函数的参数和返回值:返回的是对应值的节点,参数是节点和对应值
    3. 终止条件:如果遍历的节点是空或者就是对应数值的节点就return root。
    4. 单层递归的逻辑:创建新的变量节点result=null。如果要搜索的值比root.val小,说明在左子树,就result=函数(root.left, val);果要搜索的值比root.val大,说明在右子树,就result=函数(root.right, val);如果都没有就return result。这是个null

    3.2 迭代法思路

    1. 通过while(root!=null)遍历
    2. 如果查找的值比root.val小就向左遍历,root=root.left
    3. 如果查找的值比root.val大就向右遍历,root=root.right
    4. 如果找到了就直接return root
    5. 如果退出循环就说明没找到,就return null

    3.3 代码

    1. class Solution {
    2. // 递归,利用二叉搜索树特点,优化
    3. public TreeNode searchBST(TreeNode root, int val) {
    4. if (root == null || root.val == val) {
    5. return root;
    6. }
    7. if (val < root.val) {
    8. return searchBST(root.left, val);
    9. } else {
    10. return searchBST(root.right, val);
    11. }
    12. }
    13. }
    14. class Solution {
    15. // 迭代,利用二叉搜索树特点,优化,可以不需要栈
    16. public TreeNode searchBST(TreeNode root, int val) {
    17. while (root != null)
    18. if (val < root.val) root = root.left;
    19. else if (val > root.val) root = root.right;
    20. else return root;
    21. return null;
    22. }
    23. }

    4. LeetCode 98. 验证二叉搜索树

    4.1 思路

    1. 二叉搜索树的遍历最好是中序,因为二叉搜索树的特性“左中右”,先左再中后右就是一个有序的顺序,从小到大的顺序
    2. 递归函数的返回值和参数:返回值boolean,参数就是根节点
    3. 终止条件:如果root是null,就return true,因为空树也是二叉搜索树,同时也是完全二叉树、满二叉树、平衡二叉树
    4. 单层递归的逻辑:定义一个数组,左:函数(root.left),中:把节点值放入数组,右:函数(root.right),然后判断数组是否有序,有序说明是true否则就是false
    5. 更优解:不用创建数组。解题误区:单纯的比较左节点小于中间节点,右节点大于中间节点,这样不全面,我们要比左子树所有节点都大,比右子树所有节点都小
    6. 递归过程:返回值和参数、终止条件同上。定义一个全局变量long prev=Long.MIN_VALUE,表示最小值,因为这题会出现比Integer.MIN_VALUE还小的,只能创建个更小的才行了。
    7. 左:boolean left=函数(root.left)。如果root的值比prev大,prev就更新为root.val,prev就记录了当前节点的前一个节点的数值,因为中序遍历的root的值是递增的,prev就会持续小于root的值,这样root.val就始终比前一个节点大,如果root的值比prev小,这样就不是二叉搜索树了,就return false;
    8. 右:boolean right=函数(root.right)
    9. return left&&right。左右子树要同时符合条件这样才是符合题意的
    10. 如果这里采用前一个节点与后一个节点比较的方式的话,就需要创建一个max节点,初始化为null,如果max不为空并且max.val>=root.val,就返回false。否则就更新为root节点,这样作为记录root的前一个节点,因为root每次递归的时候都是下一个节点了,那么max就是root的前一个节点。这样可以不需要long prev初始化为最小值的方式

    4.2 代码

    1. // 简洁实现·中序遍历
    2. class Solution {
    3. private long prev = Long.MIN_VALUE;
    4. public boolean isValidBST(TreeNode root) {
    5. if (root == null) {
    6. return true;
    7. }
    8. if (!isValidBST(root.left)) {
    9. return false;
    10. }
    11. if (root.val <= prev) { // 不满足二叉搜索树条件
    12. return false;
    13. }
    14. prev = root.val;
    15. return isValidBST(root.right);
    16. }
    17. }
    18. class Solution {
    19. // 递归
    20. TreeNode max;
    21. public boolean isValidBST(TreeNode root) {
    22. if (root == null) {
    23. return true;
    24. }
    25. // 左
    26. boolean left = isValidBST(root.left);
    27. if (!left) {
    28. return false;
    29. }
    30. // 中
    31. if (max != null && root.val <= max.val) {
    32. return false;
    33. }
    34. max = root;
    35. // 右
    36. boolean right = isValidBST(root.right);
    37. return right;
    38. }
    39. }

  • 相关阅读:
    【Linux】进程 && 进程控制块PCB && 查看进程
    2023最新SSM计算机毕业设计选题大全(附源码+LW)之java宠物寄养平台设计03zp5
    全球首个“AI程序员”Deven诞生,真的能替代人类程序员吗?
    自定义NavigationBar--使用UIView进行绘制
    浅谈弧光保护在中低压电力系统中的重要性
    java毕业设计高校心理教育辅导mybatis+源码+调试部署+系统+数据库+lw
    Linux操作系统常用指令大全:系统管理篇
    (6)Mybatis-plus DML编程控制
    golang及beego框架单元测试小结
    一、 android studio安装
  • 原文地址:https://blog.csdn.net/Hsusan/article/details/133773095