算法打卡day41

今日任务:

1)198.打家劫舍

2)213.打家劫舍II

3)337.打家劫舍III

4)复习day16

198.打家劫舍

题目链接:198. 打家劫舍 - 力扣(LeetCode)

你是一个专业的小偷,计划偷窃沿街的房屋。每间房内都藏有一定的现金,影响你偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警。
给定一个代表每个房屋存放金额的非负整数数组,计算你 不触动警报装置的情况下 ,一夜之内能够偷窃到的最高金额。示例 1:
输入:[1,2,3,1]
输出:4
解释:偷窃 1 号房屋 (金额 = 1) ,然后偷窃 3 号房屋 (金额 = 3)。   偷窃到的最高金额 = 1 + 3 = 4 。示例 2:
输入:[2,7,9,3,1]
输出:12 解释:偷窃 1 号房屋 (金额 = 2), 偷窃 3 号房屋 (金额 = 9),接着偷窃 5 号房屋 (金额 = 1)。   偷窃到的最高金额 = 2 + 9 + 1 = 12。提示:
0 <= nums.length <= 100
0 <= nums[i] <= 400

文章讲解:代码随想录 (programmercarl.com)

视频讲解:动态规划,偷不偷这个房间呢?| LeetCode:198.打家劫舍哔哩哔哩bilibili

思路:

这个问题是一个典型的动态规划问题。我们可以使用动态规划来解决这个问题,其中关键是定义状态和状态转移方程。

状态定义:
定义一个数组 dp,其中 dp[i] 表示偷窃前 i 个房屋所能获取的最高金额。

状态转移方程:
考虑偷窃第 i 个房屋,有两种情况:

  1. 偷窃第 i 个房屋:如果偷窃第 i 个房屋,则不能偷窃第 i-1 个房屋,因此偷窃金额为 nums[i] + dp[i-2]
  2. 不偷窃第 i 个房屋:如果不偷窃第 i 个房屋,则偷窃金额与偷窃前 i-1 个房屋所能获取的最高金额相同,即 dp[i-1]

因此,动态转移方程为:

dp[i] = max(dp[i-1], dp[i-2] + nums[i])

初始状态:

  • 第一个房屋只有一间,因此偷窃它即可,即 dp[0] = nums[0]
  • 第二个房屋偷窃与不偷窃中选择金额较大的方案,即 dp[1] = max(nums[0], nums[1])

返回结果:
最终结果为 dp[-1],表示偷窃前所有房屋所能获取的最高金额。

class Solution:def rob(self, nums: List[int]) -> int:if not nums:  # 如果没有房屋,返回0return 0if len(nums) == 1:  # 如果只有一个房屋,返回其金额return nums[0]# 定义动态规划数组,dp[i] 表示偷窃前 i 个房屋所能获取的最高金额dp = [0] * (len(nums))dp[0] = nums[0]  # 第一个房屋只有一间,偷窃它即可dp[1] = max(nums[0], nums[1])  # 前两个房屋,选择金额较大的偷窃# 动态转移方程:偷窃当前房屋与不偷窃当前房屋的最大值for i in range(2, len(nums)):dp[i] = max(dp[i - 1], dp[i - 2] + nums[i])return dp[-1]

213.打家劫舍II

题目链接:213. 打家劫舍 II - 力扣(LeetCode)

你是一个专业的小偷,计划偷窃沿街的房屋,每间房内都藏有一定的现金。这个地方所有的房屋都 围成一圈 ,这意味着第一个房屋和最后一个房屋是紧挨着的。同时,相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警 。
给定一个代表每个房屋存放金额的非负整数数组,计算你 在不触动警报装置的情况下 ,能够偷窃到的最高金额。示例 1:
输入:nums = [2,3,2]
输出:3
解释:你不能先偷窃 1 号房屋(金额 = 2),然后偷窃 3 号房屋(金额 = 2), 因为他们是相邻的。示例 2:
输入:nums = [1,2,3,1]
输出:4
解释:你可以先偷窃 1 号房屋(金额 = 1),然后偷窃 3 号房屋(金额 = 3)。偷窃到的最高金额 = 1 + 3 = 4 。示例 3:
输入:nums = [0]
输出:0提示:
1 <= nums.length <= 100
0 <= nums[i] <= 1000

文章讲解:代码随想录 (programmercarl.com)

视频讲解:动态规划,房间连成环了那还偷不偷呢?| LeetCode:213.打家劫舍II哔哩哔哩bilibili

思路:

这个问题与之前的类似,但有一个额外的条件:房屋围成一圈。因此,我们需要特殊处理这个情况。

我们可以将这个问题拆分成两个子问题:

  1. 偷窃第一个房屋,不偷窃最后一个房屋。
  2. 偷窃最后一个房屋,不偷窃第一个房屋。

然后,我们可以使用动态规划来解决这两个子问题,最后取两个子问题中的最大值作为结果。

具体的动态规划解决方案如下:

  1. 对于第一个子问题,我们可以使用与之前类似的动态规划方法来解决。假设 dp1[i] 表示偷窃前 i 个房屋(第一个房屋被偷窃)所能获取的最高金额,则有以下状态转移方程:

    dp1[i] = max(dp1[i-1], dp1[i-2] + nums[i]) 其中 dp1[0] = nums[0],dp1[1] = max(nums[0], nums[1])
  2. 对于第二个子问题,同样使用动态规划方法。假设 dp2[i] 表示偷窃前 i 个房屋(最后一个房屋被偷窃)所能获取的最高金额,则有以下状态转移方程:

    dp2[i] = max(dp2[i-1], dp2[i-2] + nums[i]) 其中 dp2[0] = 0dp2[1] = nums[1]

最终,我们可以将两个子问题的最终结果进行比较,取其中较大的值作为最终结果。

class Solution:def rob(self, nums: List[int]) -> int:if not nums:  # 如果没有房屋,返回0return 0if len(nums) == 1:  # 如果只有一个房屋,返回其金额return nums[0]# 计算偷窃第一个房屋,不偷窃最后一个房屋的最高金额res1 = self.rob_range(nums,0,len(nums)-2)# 计算偷窃最后一个房屋,不偷窃第一个房屋的最高金额res2 = self.rob_range(nums,1,len(nums)-1)# 取两个子问题的最大值作为最终结果return max(res1,res2)# 动态规划解决子问题def rob_range(self,nums: List[int], start: int, end: int) -> int:if end == start:return nums[start]# 创建一个数组用于存储每个房屋偷窃时的最高金额dp = [0]*(end-start+1)# 初始化前两个房屋的最高金额dp[0] = nums[start]dp[1] = max(nums[start],nums[start+1])for i in range(2, end - start + 1):# 在偷窃当前房屋和不偷窃当前房屋之间选择金额较大的方案dp[i] = max(dp[i-1],dp[i-2]+nums[start+i])return dp[-1]

337.打家劫舍III

题目链接:337. 打家劫舍 III - 力扣(LeetCode)

在上次打劫完一条街道之后和一圈房屋后,小偷又发现了一个新的可行窃的地区。这个地区只有一个入口,我们称之为“根”。 除了“根”之外,每栋房子有且只有一个“父“房子与之相连。一番侦察之后,聪明的小偷意识到“这个地方的所有房屋的排列类似于一棵二叉树”。 如果两个直接相连的房子在同一天晚上被打劫,房屋将自动报警。
计算在不触动警报的情况下,小偷一晚能够盗取的最高金额。示例 1:
输入: root = [3,2,3,null,3,null,1]
输出: 7
解释: 小偷一晚能够盗取的最高金额 3 + 3 + 1 = 7示例 2:
输入: root = [3,4,5,1,3,null,1]
输出: 9
解释: 小偷一晚能够盗取的最高金额 4 + 5 = 9提示:
树的节点数在 [1, 104] 范围内
0 <= Node.val <= 104

文章讲解:代码随想录 (programmercarl.com)

视频讲解:动态规划,房间连成树了,偷不偷呢?| LeetCode:337.打家劫舍3哔哩哔哩bilibili

思路:

在这个问题中,我们有一棵二叉树,表示小偷可以盗取的房屋。每个节点表示一个房屋,节点的值表示房屋内的现金。树中的相邻节点表示相邻的房屋,如果相邻的房屋在同一晚上被打劫,系统会自动报警。

我们的目标是计算在不触发警报的情况下,小偷一晚能够盗取的最高金额。

解决这个问题的一种方法是使用动态规划。我们可以定义一个递归函数rob(root),其中 root 是当前节点。对于每个节点,我们有两种选择:

  1. 盗取当前节点和其孙子节点的价值,然后考虑其孙子节点的左右孩子。
  2. 不盗取当前节点,而是考虑其两个子节点。

我们可以通过递归地计算这两种选择来获得最大金额。但是,这样的方法可能会导致大量的重复计算。为了避免这种情况,我们可以使用记忆化搜索或动态规划来优化递归解法。在动态规划方法中,我们可以定义一个状态数组 dp,其中 dp[node] 表示盗取以 node 为根节点的子树所能获得的最大金额。然后,我们可以根据状态转移方程来计算 dp[node]。

具体地,对于每个节点 node,我们有两种选择:

  1. 如果我们盗取了当前节点 node,则不能盗取其孩子节点。因此,最大金额为 node.val + dp[node.left.left] + dp[node.left.right] + dp[node.right.left] + dp[node.right.right]。
  2. 如果我们不盗取当前节点 node,则可以选择盗取其左孩子和右孩子。因此,最大金额为 dp[node.left] + dp[node.right]。

最后,我们返回根节点的最大金额,即 dp[root]。

这样,我们可以通过动态规划来高效地解决这个问题,确保小偷在不触发警报的情况下,一晚能够盗取的最高金额。

# Definition for a binary tree node.
class TreeNode:def __init__(self, val=0, left=None, right=None):self.val = valself.left = leftself.right = rightclass Solution:def rob(self, root: TreeNode) -> int:# 调用辅助函数并返回根节点的最大金额return max(self.dfs(root))# 定义一个辅助函数来计算盗取以root为根节点的子树所能获得的最大金额def dfs(self, node):if not node:return 0, 0  # 返回不盗取当前节点和盗取当前节点的最大金额# 递归计算左右子树的最大金额left_no_rob, left_rob = self.dfs(node.left)right_no_rob, right_rob = self.dfs(node.right)# 不盗取当前节点时,可以选择盗取左右子树的根节点或者不盗取左右子树的根节点no_rob = max(left_no_rob, left_rob) + max(right_no_rob, right_rob)# 盗取当前节点时,不能盗取左右子树的根节点rob = node.val + left_no_rob + right_no_robreturn no_rob, robroot = TreeNode(3,2,3)
root.left = TreeNode(2)
root.right = TreeNode(3)
root.left.right = TreeNode(3)
root.right.right = TreeNode(1)obj = Solution()print(obj.rob(root))

感想:这题有点变化,用到了二叉树,这里要先算子节点的最大值,再算父节点,所以这题要采用后序遍历。

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

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

相关文章

【hive】transform脚本

文档地址&#xff1a;https://cwiki.apache.org/confluence/display/Hive/LanguageManualTransform 一、介绍二、实现1.脚本上传到本地2.脚本上传到hdfs 三、几个需要注意的点1.脚本名不要写全路径2.using后面语句中&#xff0c;带不带"python"的问题3.py脚本Shebang…

LNMP部署wordpress

1.环境准备 总体架构介绍 序号类型名称外网地址内网地址软件02负载均衡服务器lb0110.0.0.5192.168.88.5nginx keepalived03负载均衡服务器lb0210.0.0.6192.168.88.6nginx keepalived04web服务器web0110.0.0.7192.168.88.7nginx05web服务器web0210.0.0.8192.168.88.8nginx06we…

基于Springboot的校园生活服务平台(有报告)。Javaee项目,springboot项目。

演示视频&#xff1a; 基于Springboot的校园生活服务平台&#xff08;有报告&#xff09;。Javaee项目&#xff0c;springboot项目。 项目介绍&#xff1a; 采用M&#xff08;model&#xff09;V&#xff08;view&#xff09;C&#xff08;controller&#xff09;三层体系结构…

shell脚本-监控系统内存和磁盘容量

监控内存和磁盘容量除了可以使用zabbix监控工具来监控&#xff0c;还可以通过编写Shell脚本来监控。 #! /bin/bash #此脚本用于监控内存和磁盘容量&#xff0c;内存小于500MB且磁盘容量小于1000MB时报警#提取根分区剩余空间 disk_size$(df / | awk /\//{print $4})#提取内存剩…

《机器学习算法面试宝典》重磅发布!

我们经常会组织场算法岗技术&面试讨论会&#xff0c;会邀请了一些互联网大厂朋友、今年参加社招和校招面试的同学。 针对新手如何入门算法岗、该如何准备面试攻略、面试常考点等热门话题进行了深入的讨论。 基于讨论和经验总结&#xff0c;历时半年的梳理和修改&#xff…

eNSP-浮动静态路由配置

ip route-static 192.168.1.0 24 192.168.3.2 preference 60 #设置路由 目标网络地址 和 下一跳地址 preference值越大 优先级越低 一、搭建拓扑结构 二、主机配置 pc1 pc2 三、配置路由器 1.AR1路由器配置 <Huawei>sys #进入系统视图 [Huawei]int g0/0/0 #进入接…

详解面向对象-类和对象

1.面向对象与面向过程的区别 ①面向过程 &#xff1a;关注点是在实现功能的步骤上面&#xff0c;就是分析出解决问题所需要的步骤&#xff0c;让后函数把这些步骤一步一步实现&#xff0c;使用的时候一个一个依次调用就可以。对于简单的流程是适合面向过程的方式进行的&#x…

C++:set和map的介绍

目录 关联式容器 键值对 set介绍&#xff1a; set的模板参数列表 set的双向迭代器&#xff1a; insert的使用和set的特性&#xff1a; set的删除&#xff1a; set的find&#xff1a; lower_bound 、 upper_bound&#xff1a; multiset&#xff1a; map介绍&#xff…

解决windows中的WSL Ubuntu子系统忘记root密码和用户密码问题

1、以管理员身份运行PowerShell 2、在powershell中执行wsl.exe --user root wsl.exe --user root如果出现了上面的报错&#xff0c;则需要运行步骤3、4&#xff0c;然后在执行步骤5改密码&#xff0c;如果没有出错&#xff0c;请直接跳到第5步改密码操作&#xff01;&#xff…

第11章 软件工程

这里写目录标题 1.软件过程1.1能力成熟度模型(CMM)1.2能力成熟度模型集成(CMMI)1.3瀑布模型(线性顺序)1.4增量模型1.5演化模型1.5.1原型模型1.5.2螺旋模型 1.6喷泉模型1.7统一过程(UP)模型 2.敏捷方法3.系统设计4.系统测试4.1单元测试(模块测试)4.2集成测试4.3黑盒测试(功能测试…

HNU-人工智能-实验1-A*算法

人工智能-实验1 计科210x 甘晴void 一、实验目的 掌握有信息搜索策略的算法思想&#xff1b; 能够编程实现搜索算法&#xff1b; 应用A*搜索算法求解罗马尼亚问题。 二、实验平台 课程实训平台https://www.educoder.net/shixuns/vgmzcukh/challenges 三、实验内容 3.…

如何将数据导入python

Python导入数据的三种方式&#xff1a; 1、通过标准的Python库导入CSV文件 Python提供了一个标准的类库CSV文件。这个类库中的reader()函数用来导入CSV文件。当CSV文件被读入后&#xff0c;可以利用这些数据生成一个NumPy数组&#xff0c;用来训练算法模型。 from csv import…

详细介绍如何使用YOLOv9 在医疗数据集上进行实例分割-含源码+数据集下载

深度学习彻底改变了医学图像分析。通过识别医学图像中的复杂模式,它可以帮助我们解释有关生物系统的重要见解。因此,如果您希望利用深度学习进行医疗诊断,本文可以成为在医疗数据集上微调YOLOv9 实例分割的良好起点。 实例分割模型不是简单地将区域分类为属于特定细胞类型,…

基于Springboot的校园竞赛管理系统(有报告)。Javaee项目,springboot项目。

演示视频&#xff1a; 基于Springboot的校园竞赛管理系统&#xff08;有报告&#xff09;。Javaee项目&#xff0c;springboot项目。 项目介绍&#xff1a; 采用M&#xff08;model&#xff09;V&#xff08;view&#xff09;C&#xff08;controller&#xff09;三层体系结构…

stl容器 string类的基本操作

目录 一.string类的构造 二.string类的输出 1.传统字符串输出 2.通过迭代器进行输出 ​编辑 3.C11标准的范围for输出加auto推导类型 三.string类的各种迭代器 begin(&#xff09;和end() 利用迭代器遍历输出 利用迭代器修改字符串的字符 rbgin()和rend() 利用迭代器遍…

创造价值与回报:创业者的思维格局与商业智慧

在纷繁复杂的商业世界中&#xff0c;有一种信念始终贯穿于无数创业者的心中——那就是创造价值。张磊的这句“只要不断地创造价值&#xff0c;迟早会有回报”道出了创业者的核心思维格局和商业智慧。本文将从创业者的角度&#xff0c;探讨创造价值的重要性&#xff0c;以及如何…

5.Spring Security-web权限方案

设置登录的用户名和密码 1.通过配置文件设置用户名密码 spring:security:user:name: xiankejinpassword: 123456 如果没有以上配置&#xff0c;那么就会在后台生成一个随机密码&#xff0c;用户名固定位user。 2.通过配置类设置用户名密码 Configuration public class Sec…

为什么说虚拟化技术是现代网络安全的重要组成部分?

虚拟化技术是一种对计算机资源的抽象和资源管理技术&#xff0c;将电脑的各种实体资源&#xff08;CPU、内存、磁盘空间、网络适配器等&#xff09;予以抽象、转换后呈现出来&#xff0c;并可供分割、组合为一个或多个电脑配置环境。今天德迅云安全带您了解为什么虚拟化技术能成…

翻译: 什么是ChatGPT 通过图形化的方式来理解 Transformer 架构 深度学习一

合集 ChatGPT 通过图形化的方式来理解 Transformer 架构 翻译: 什么是ChatGPT 通过图形化的方式来理解 Transformer 架构 深度学习一翻译: 什么是ChatGPT 通过图形化的方式来理解 Transformer 架构 深度学习二翻译: 什么是ChatGPT 通过图形化的方式来理解 Transformer 架构 深…

基于AT89C51单片机的温度上下限自动控制检报警设计

点击链接获取Keil源码与Project Backups仿真图: https://download.csdn.net/download/qq_64505944/89247694?spm=1001.2014.3001.5501 C 源码+仿真图+毕业设计+实物制作步骤+06 题 目 基于单片机的温度检测调节系统设计 姓 名 学 号 专业班级 指导教师 年 月 日 任务书 …