代码随想录算法训练营第四十七天 | LeetCode 198. 打家劫舍、213. 打家劫舍 II、337. 打家劫舍 III

news2025/1/12 5:55:56

代码随想录算法训练营第四十七天 | LeetCode 198. 打家劫舍、213. 打家劫舍 II、337. 打家劫舍 III

文章链接:打家劫舍 打家劫舍 II 打家劫舍 III
视频链接:打家劫舍 打家劫舍 II 打家劫舍 III

1. LeetCode 198. 打家劫舍

1.1 思路

  1. 我们要去偷钱,但相邻房间不能偷,求最后偷的最大金额。其实我们对于当前房间偷不偷是取决于前一个和前前一个房间的,是一个递推的关系。
  2. dp 数组及其下标的含义:dp[i] 考虑下标 i(包含下标 i),所能偷的最多的金额为 dp[i],最终结果在 dp[nums.length-1]。注意我们是考虑,考虑的仅仅是遍历的范围,取不取由递推公式决定
  3. 递推公式:偷 i 和不偷 i。偷 i 就是只能考虑前前一个房间,即 dp[i-2]+nums[i],i-2 之前的范围加上 i 就是我们的考虑范围。不偷 i 就是考虑前一个房间,即 dp[i-1],i-1 之前的范围就是我们的考虑范围。因此递推公式:dp[i]=Math.max(dp[i-2]+nums[i],dp[i-1])
  4. dp 数组的初始化:根据递推公式,我们的基础就是 dp[0] 和 dp[1],dp[0] 就只能是偷 nums[0],dp[1] 是考虑下标 1 之前的包括下标 1,1 和 0 两个位置取最大值 dp[1]=Math.max(nums[1],nums[0]),其余下标初始化为 0 即可,不影响
  5. 遍历顺序:根据递推公式,就是从前往后比那里的 for(int i=2;i<nums.length;i++)

1.2 代码

// 动态规划
class Solution {
	public int rob(int[] nums) {
		if (nums == null || nums.length == 0) return 0;
		if (nums.length == 1) return nums[0];

		int[] dp = new int[nums.length];
		dp[0] = nums[0];
		dp[1] = Math.max(dp[0], nums[1]);
		for (int i = 2; i < nums.length; i++) {
			dp[i] = Math.max(dp[i - 1], dp[i - 2] + nums[i]);
		}

		return dp[nums.length - 1];
	}
}

2. LeetCode 213. 打家劫舍 II

2.1 思路

  1. 本题和198. 打家劫舍的区别就是环了,之前是一个线性的数组,本题是把这个数组连成环了,首尾相连,其余的都是相同的。关于连成环首尾怎么取,我们可以分成下面三种情况:
  2. 情况 1:首尾都不取,只取中间部分,对于数组是否连成环跟这种情况没有关系,就相当于是线性数组了,直接和我们在198. 打家劫舍处理方式一样
  3. 情况 2:考虑首元素不考虑尾元素,就相当于默认数组没有尾元素了,这样对于数组是否连成环也没有关系了
  4. 情况 3:不考虑首元素考虑尾元素,就相当于默认数组没有首元素了,这样对于数组是否连成环也没有关系了
  5. 关于连成环就是分成了以上三种情况了,但是情况 2 和 3 是包含 1 的。情况 1 是考虑中间部分,情况 2 是考虑首+中间部分,情况 3 是考虑尾+中间部分。注意我们是考虑,不是一定要取,考虑的仅仅是遍历的范围,取不取是由递推公式决定的。因此我们只要求情况 2 的最优解和情况 3 的最优解,两者取最大值即可。我们可以把情况 2 和情况 3 分别传到198. 打家劫舍函数里得到这个线性数组的最大值,两者再取最大值

2.2 代码

class Solution {
    public int rob(int[] nums) {
        if (nums == null || nums.length == 0)
            return 0;
        int len = nums.length;
        if (len == 1)
            return nums[0];
        return Math.max(robAction(nums, 0, len - 1), robAction(nums, 1, len));
    }

    int robAction(int[] nums, int start, int end) {
        int x = 0, y = 0, z = 0;
        for (int i = start; i < end; i++) {
            y = z;
            z = Math.max(y, x + nums[i]);
            x = y;
        }
        return z;
    }
}

3. LeetCode 337. 打家劫舍 III

3.1 思路

  1. 本题和前面不一样的是我们是在一个二叉树上偷,要求也是相邻节点不能偷,就相当于是树形 dp,因此也用到了之前的递归三部曲
  2. dp 数组及其下标的含义:每个节点只有两个状态,偷与不偷,用一个长度为 2 的 dp 数组就可以表示了,dp[0]=不偷,dp[1]=偷。因为我们在遍历二叉树的过程中是通过递归遍历的,系统栈会保存每一层递归里的参数,每一层递归都有一个长度为 2 的 dp 数组,当前层 dp 数组就是表示当前层所遍历这个节点的状态,dp[0] 就是不偷所得的最大金额,dp[1] 就是偷所得的最大金额。而我们是通过后序遍历从底向上遍历的,最后就是根节点偷与不偷两个状态取最大值
  3. 递归函数的参数和返回值:返回值是一个 dp 数组,一维的,长度为 2,参数是 root。我们是通过一个数组来接收这个函数的返回值的。最终是 return Math.max(数组 [0],数组 [1])两个状态取最大值
  4. 递归函数的终止条件:if(root==null)此时偷与不偷的最大金额都是 0,因为是空节点
  5. 遍历顺序:偷与不偷取一个最大值。偷当前节点,左右孩子就不能偷了 int value1=root.val+leftdp[0]+rightdp[0]。这里的leftdp 和rightdp 就是我们通过后序遍历从底往上推的过程得到的,因此要在上面定义 dp 数组 leftdp 和rightdp 通过递归运算得到 leftdp=递归函数(root.left),rightdp=递归函数(root.right),这样就得到了左右孩子偷与不偷的最大值,因此就能得到当前节点偷与不偷的最大值,当前节点偷了那左右孩子就不能偷 int value1=root.val+leftdp[0]+rightdp[0],当前节点不偷那左右孩子考虑能偷,偷不偷取决于左右孩子偷与不偷的最大值是什么,dp[0] 和 dp[1] 哪个大就取哪个 int value2=Math.max(leftdp[0],leftdp[1])+Math.max(rightdp[0],rightdp[1]),最终 return value2,value1 组成的数组,注意两个的位置。并且我们上面的逻辑是"左右中",即后序遍历的逻辑
    在这里插入图片描述

3.2 代码

// 3.状态标记递归
    // 执行用时:0 ms , 在所有 Java 提交中击败了 100% 的用户
    // 不偷:Max(左孩子不偷,左孩子偷) + Max(又孩子不偷,右孩子偷)
    // root[0] = Math.max(rob(root.left)[0], rob(root.left)[1]) +
    // Math.max(rob(root.right)[0], rob(root.right)[1])
    // 偷:左孩子不偷+ 右孩子不偷 + 当前节点偷
    // root[1] = rob(root.left)[0] + rob(root.right)[0] + root.val;
    public int rob3(TreeNode root) {
        int[] res = robAction1(root);
        return Math.max(res[0], res[1]);
    }

    int[] robAction1(TreeNode root) {
        int res[] = new int[2];
        if (root == null)
            return res;

        int[] left = robAction1(root.left);
        int[] right = robAction1(root.right);

        res[0] = Math.max(left[0], left[1]) + Math.max(right[0], right[1]);
        res[1] = root.val + left[0] + right[0];
        return res;
    }
}

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

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

相关文章

python使用memory_profiler分析代码运行内存占用

memory_profile memory_profiler源码仓库 安装 pip install memory_profiler 使用 请参考以下文章,写的很详细 【精选】Python代码优化工具——memory_profiler_被Python玩的Kenny的博客-CSDN博客 本文要增加介绍的是API使用 目录结构 |--my.py |--tests | |-- test_m…

设计模式之保护性暂停

文章目录 1. 定义2. 实现保护性暂停模式3. Join原理4. 保护性暂停模式的扩展 1. 定义 即Guarded Suspension&#xff0c;用在一个线程等待另一个线程的执行结果。 有一个结果需要从一个线程传递给另一个线程&#xff0c;让他们关联到同一个GuarderObject&#xff08;这就是保…

快速教程|如何在 AWS EC2上使用 Walrus 部署 GitLab

Walrus 是一款基于平台工程理念的开源应用管理平台&#xff0c;致力于解决应用交付领域的深切痛点。借助 Walrus 将云原生的能力和最佳实践扩展到非容器化环境&#xff0c;并支持任意应用形态统一编排部署&#xff0c;降低使用基础设施的复杂度&#xff0c;为研发和运维团队提供…

汽车生产RFID智能制造设计解决方案与思路

汽车行业需求 汽车行业正面临着快速变革&#xff0c;传统的汽车制造方式正在向柔性化、数字化、自动化和数据化的智能制造体系转变&#xff0c;在这个变革的背景下&#xff0c;汽车制造企业面临着物流、生产、配送和资产管理等方面的挑战&#xff0c;为了应对这些挑战&#xf…

为什么亚马逊的轻量应用服务器这么受欢迎 | 个人体验 | 优势所在

文章目录 &#x1f33a;前言⭐什么是轻量应用服务器&#x1f6f8;特点 &#x1f384;亚马逊轻量应用服务器体验如何&#x1f339;亚马逊轻量应用服务器的优势 &#x1f33a;前言 作为一为开发者&#xff0c;我们要开发部署一个自己的网站&#xff0c;要选择一个性能好的服务器…

Mercury性能测试模板

xxxxxxxxxx 性能测试报告 2023年11月8日 目 录 1 前言 1 1第一章XXXXXXXX核心业务系统性能测试概述 1 1.1 被测系统定义 1 1.1.1 功能简介 1 1.1.2 性能测试指标 2 1.2 系统结构及流程 2 1.2.1 系统总体结构 2 1.2.2 功能模块描述 3 1.2.3 业务…

简单免费工具帮你恢复Windows中已删除的文件!

​如果你在Windows上丢失了文件&#xff0c;有许多工具可以找到并恢复它们&#xff0c;但WinFR界面版既免费又好用&#xff0c;借助该软件你可以轻松恢复丢失的文件。 Microsoft提供了一款免费的工具&#xff0c;可以帮你找到并恢复因意外删除或磁盘或软件问题导致丢…

Nginx实现tcp代理并实现TLS加密

Nginx源码编译 关于nginx的搭建配置具体参考笔者之前的一篇文章&#xff1a;实时流媒体服务器搭建试验&#xff08;nginxrtmp&#xff09;_如何在线测试流媒体rtmp搭建成功了吗-CSDN博客中的前半部分&#xff1b;唯一变化的是编译参数&#xff08;添加stream模块并添加其对应ss…

竞赛选题 深度学习手势识别 - yolo python opencv cnn 机器视觉

文章目录 0 前言1 课题背景2 卷积神经网络2.1卷积层2.2 池化层2.3 激活函数2.4 全连接层2.5 使用tensorflow中keras模块实现卷积神经网络 3 YOLOV53.1 网络架构图3.2 输入端3.3 基准网络3.4 Neck网络3.5 Head输出层 4 数据集准备4.1 数据标注简介4.2 数据保存 5 模型训练5.1 修…

java中拼接“

params " " paramName "\"" paramValue "\"";

为什么说数据安全运维难?有好用的数据安全运维平台吗?

随着息技术的快速发展&#xff0c;不少企业在实行数字化转型&#xff0c;同时也面临着越来越多的数据安全运维挑战。不少企业都觉得数据安全运维难&#xff0c;都在找好用的数据安全运维平台。今天我们就来聊聊为什么说数据安全运维难&#xff1f;以及是否有好用的数据安全运维…

指纹浏览器有什么用?盘点指纹浏览器八大应用场景

在网络世界中&#xff0c;浏览器指纹一词早已耳熟能详。比如你用的浏览器类型、设备信息&#xff0c;甚至是你的操作习惯等等这些浏览器指纹信息可以让网站轻松识别出你的身份&#xff0c;把你的信息发送给第三方的广告商&#xff0c;然后匹配你的情况进行广告营销。 即使在清除…

广州小程序开发公司怎么找?

在当今的数字化时代&#xff0c;小程序已经成为了企业拓展业务、提升品牌影响力的重要工具。广州作为国内经济中心之一&#xff0c;拥有众多的企业和商家&#xff0c;因此对于小程序开发的需求也日益增长。那么&#xff0c;如何在广州找到一家专业、可靠的小程序开发公司呢&…

六种最常见的软件供应链攻击

软件供应链攻击已成为当前网络安全领域的热点话题&#xff0c;其攻击方式的多样性和复杂性使得防御变得极为困难。以下我们整理了六种常见软件供应链攻击方法及其典型案例&#xff1a; 软件供应链攻击已成为当前网络安全领域的热点话题&#xff0c;其攻击方式的多样性和复杂性…

TSN协议解读系列 | (3) Qci:说书唱戏劝人方

你是谁&#xff1f;你从哪里来&#xff1f;你到哪里去&#xff1f;这是柳国旅问的最多的三个问题。他总在想&#xff0c;上辈子的自己一定是一个哲学家&#xff0c;不然也不会和这三个问题的关系如此密切。他站的笔挺&#xff0c;耳边是蝉鸣&#xff0c;眼前是蓝天&#xff0c;…

定岗定编设计:企业职能部门定岗定编设计项目成功案例

一、客户背景及现状分析 某大型车辆公司隶属于某央企集团&#xff0c;建于20世纪60年代&#xff0c;是中国高速、重载、专用铁路车辆生产经营的优势企业&#xff0c;轨道车辆制动机研发制造的主导企业&#xff0c;是隶属于国内最大的轨道交通设备制造上市企业的骨干二级公司。公…

JVM虚拟机:垃圾回收器之ParNew(年轻代)

本文重点 在前面的课程中,我们学习了新生代的垃圾回收器PS,本文我们将学习新生代的另一个垃圾回收器ParNew。 工作状态 这个垃圾回收器使用多线程进行垃圾回收,在垃圾回收时,会STW(stop-the-world)暂停其它所有的工作线程直到它的收集结束,如下所示: 配置 -XX:+UserP…

南昌大学漏洞报送证书

获取来源&#xff1a;edusrc&#xff08;教育漏洞报告平台&#xff09; url&#xff1a;https://src.sjtu.edu.cn/ 兑换价格&#xff1a;20金币 获取条件&#xff1a;南昌大学任意中危或以上级别漏洞

Matlab论文插图绘制模板第125期—特征渲染的三维气泡图

在之前的文章中&#xff0c;分享了很多Matlab三维气泡图的绘制模板&#xff1a; 进一步&#xff0c;再来分享一下特征渲染的三维气泡图。 先来看一下成品效果&#xff1a; 特别提示&#xff1a;本期内容『数据代码』已上传资源群中&#xff0c;加群的朋友请自行下载。有需要的…

编程系统化教程目录,中文编程工具下载

图下图是编程工具界面&#xff0c;其构件板构件非常丰富。想学编程&#xff0c;可以点击最下方卡片—— 软件下载——免费自由版软件下载及教程&#xff0c;了解详细资讯。 编程系统化教程视频课程总目录 链接&#xff0c;点击下方链接进入 https://jywxz.blog.csdn.net/art…