• 代码随想录训练营day48, 打家劫舍系列问题


    打家劫舍

    可以体会一下, 当前房屋偷不偷取决于前一个房屋和前两个房屋是否被偷了, 所以这种依赖关系就是dp的递推公式

    1. dp[i]: 考虑下标i以内的房屋, 最多可以偷窃的金额为dp[i]
    2. 递推公式: 决定的因素就是第i房间是偷还是不偷, 然后比较max
      1. 偷i的话, 那么dp[i] = dp[i-2] + nums[i] (不会偷上一个, 然后加上i)
      2. 不偷i的话, 那么dp[i] = dp[i - 1]
    3. 初始化: 可以从递推公式看出, 基础就是dp[0]和dp[1], 从定义可以看出dp[0]一定是nums[0], dp[1]一定是nums[0]和nums[1]的最大值(0-最多偷窃的金额为nums[0]; 1-只有两间房, 偷金额大的那个)
    4. 遍历顺序: 从前到后
    5. 最后dp数组的最后一个值就是结果, 所以其实是length-1的位置
    1. class Solution {
    2. public int rob(int[] nums) {
    3. //前面要多剪剪枝
    4. if(nums.length == 0 || nums == null){
    5. return 0;
    6. }
    7. if(nums.length == 1){
    8. return nums[0];
    9. }
    10. //开始dp数组创建和初始化
    11. int[] dp = new int[nums.length + 1];
    12. dp[0] = nums[0];
    13. dp[1] = Math.max(nums[0], nums[1]);
    14. //开始从前向后遍历
    15. for(int i = 2; i < nums.length; i++){
    16. dp[i] = Math.max(dp[i - 1], dp[i - 2] + nums[i]);
    17. }
    18. //找的是最后位置, 所以减一
    19. return dp[nums.length - 1];
    20. }
    21. }

    打家劫舍II

    唯一的区别就是变成环了

    成环主要考虑三种情况:

    1. 考虑不包含首尾元素
    2. 考虑包含首元素, 不包含尾元素
    3. 考虑包含尾元素, 不包含首元素

    (而其实情况二和三已经包含了情况一, 所以比较二三就行了)

    可以先看一下完整版代码: 注意关于start和end的变量设置

    1. class Solution {
    2. public int rob(int[] nums) {
    3. //还是一样的剪枝
    4. int len = nums.length;
    5. if(nums == null || len == 0){
    6. return 0;
    7. }
    8. if(len == 1){
    9. return nums[0];
    10. }
    11. //开始判断两种条件
    12. //首先是只考虑头
    13. int result1 = robRange(nums, 0, len - 2);
    14. //然后是只考虑尾
    15. int result2 = robRange(nums, 1, len - 1);
    16. return Math.max(result1, result2);
    17. }
    18. public int robRange(int[] nums, int start, int end){
    19. //这里还要判断一下, 两个数相等, 就去第一个, 否则后面的max会出错
    20. if(end == start) return nums[start];
    21. //这里还是打家劫舍逻辑
    22. int[] dp = new int[nums.length + 1];
    23. dp[start] = nums[start];
    24. dp[start + 1] = Math.max(nums[start], nums[start + 1]);
    25. for(int i = start + 2; i <=end; i++){
    26. dp[i] = Math.max(dp[i - 2] + nums[i], dp[i - 1]);
    27. }
    28. return dp[end];
    29. }
    30. }

    简化版: 动态方程就是1. 偷了n(dp[n + 1]就是dp[n]) 2. 没偷n(dp[n + 1]就是dp[n-1] + num)

    这里的思想就是用n+1的位置

    1. class Solution {
    2. public int rob(int[] nums) {
    3. int len = nums.length;
    4. if(len == 0 || nums == null){
    5. return 0;
    6. }
    7. if(len == 1){
    8. return nums[0];
    9. }
    10. int result1 = robHelper(nums, 0, len - 1);
    11. int result2 = robHelper(nums, 1, len);
    12. return Math.max(result1, result2);
    13. }
    14. public int robHelper(int[] nums, int start, int end){
    15. if(start == end) return nums[start];
    16. int pre = 0, cur = 0, temp = 0;
    17. //利用滚动数组来模拟dp
    18. for(int i = start; i < end; i++){
    19. temp = cur;
    20. cur = Math.max(pre + nums[i], cur);
    21. pre = temp;
    22. }
    23. return cur;
    24. }
    25. }

    打家劫舍III

    这次就变成了二叉树

    回想起之前的遍历方式, 这里是后序, 因为通过递归函数的返回值来做下一步计算

    若抢了当前节点, 两个孩子就不能动, 如果没抢当前节点, 就可以考虑抢左右孩子

    树形dp, 结合递归三部曲和动规五部曲

    1. 确定返回值: 就是返回dp数组, 长度为2.下标为0记录不偷该节点所得到的最大金钱, 下标为1记录偷该节点所得到的最大金钱

    2. 确定终止条件: 如果遇到空节点时, 偷或不偷都是0, 所以return

    3. 遍历顺序: 使用后序

    4. 单层递归的逻辑:

      1. 如果偷当前, 两个孩子就不能偷, cur.val + left[0] + right [0]
      2. 如果不偷, 就可以偷孩子 max(left[0], left[1]) + max(right[0], right[1])

      最终就是return(val2, val1), 不偷得到的最大钱和偷得到的

    1. class Solution {
    2. public int rob(TreeNode root) {
    3. int[] res = robRecur(root);
    4. return Math.max(res[0], res[1]);
    5. }
    6. private int[] robRecur(TreeNode root){
    7. //穿件一个长度为2的数组
    8. int[] dp = new int[2];
    9. //先确定终止条件
    10. if(root == null){
    11. return dp;
    12. }
    13. //后序遍历
    14. int[] left = robRecur(root.left);
    15. int[] right = robRecur(root.right);
    16. //单层逻辑
    17. //首先是不偷的情况, 那么就是偷孩子,
    18. dp[0] = Math.max(left[0], left[1]) + Math.max(right[0], right[1]);
    19. //然后是偷的情况, 所以孩子就不能偷
    20. dp[1] = root.val + left[0] + right[0];
    21. return dp;
    22. }
    23. }

  • 相关阅读:
    从零玩转之JPOM自动化部署本地构建 + SSH 发布 java 项目
    【CCF会议期刊推荐】中国计算机协会(CCF)推荐国际学术期刊/会议(计算机体系结构/并行与分布计算/存储系统)
    【后端】git与python的结合使用
    Selenium入门
    编译搭建ngrok服务实现内网穿透
    Go:Signal信号量的简介与实践(优雅的退出)
    aws亚马逊云:置以使用 Amazon EC2!!!
    查看使用Android API接口创建的AppLinking链接的分析数据
    Java 编程问题:一、字符串、数字和数学
    计算机毕业设计之java+springboot基于vue的地方美食分享网站
  • 原文地址:https://blog.csdn.net/Southside3amurai/article/details/128090887