【附源码】Python :打家劫舍

embedded/2024/10/15 20:18:34/

系列文章目录

Python 算法学习:打家劫舍问题


文章目录

  • 系列文章目录
  • 一、算法需求
  • 二、解题思路
  • 三、具体方法+源码
    • 方法1:动态规划(自底向上)
    • 方法2:动态规划(自顶向下)
    • 方法3:优化的动态规划
    • 方法4:递归
  • 总结


一、算法需求

“打家劫舍”问题是一个经典的动态规划问题,通常用来描述一个小偷在一条街上偷窃房屋的场景。每间房屋都有一定数量的现金,小偷需要决定偷哪些房屋以最大化他的收益。但是,小偷面临一个限制:如果两间相邻的房屋在同一晚上被偷,那么防盗系统会触发报警。因此,小偷不能偷窃相邻的房屋。


二、解题思路

动态规划: 定义一个数组 dp,其中 dp[i] 表示到第 i 间房屋为止能偷到的最大金额。状态转移方程是 dp[i] = max(dp[i-1], dp[i-2] + nums[i]),表示可以选择偷当前房子(前提是不偷前一个房子)或者不偷当前房子(延续前一个房子的决策)。

贪心算法: 虽然不总是最优,但可以作为一种尝试。在每一步选择当前能获得的最大金额,而不考虑未来的房子。

递归: 通过递归函数模拟决策过程,考虑偷或不偷当前房子,并取两种选择中的最大值。

优化空间: 使用两个变量代替数组,减少空间复杂度。


三、具体方法+源码

方法1:动态规划(自底向上)

状态定义:dp[i] 表示到第 i 个房子为止能偷到的最大金额。

计算过程:

dp[0] = 2(只考虑第一个房子)
dp[1] = max(2, 7) = 7(考虑第一个和第二个房子)
dp[2] = max(7, 2+9) = 9(考虑第二个和第三个房子)
dp[3] = max(9, 7+3) = 10(考虑第三个和第四个房子)
dp[4] = max(10, 9+1) = 12(考虑第四个和第五个房子)
结果:12

代码如下:

python">def rob1(nums):if not nums:return 0if len(nums) == 1:return nums[0]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]# 测试
nums = [2, 7, 9, 3, 1]
print("最高金额:", rob1(nums))

方法2:动态规划(自顶向下)

计算过程:

从 rob(0) 开始
rob(1) = max(rob(2), rob(3)) = max(7, 9) = 9
rob(2) = max(rob(3), rob(4) + 2) = max(9, 10) = 10
rob(3) = max(rob(4), rob(5) + 3) = max(9, 12) = 12
rob(4) = max(rob(5), rob(6) + 1) = max(7, 12) = 12
结果:12

代码如下:

python">def rob2(nums):memo = {}def rob(i):if i >= len(nums):return 0if i in memo:return memo[i]memo[i] = max(rob(i+1), nums[i] + rob(i+2))return memo[i]return rob(0)# 测试
nums = [2, 7, 9, 3, 1]
print("最高金额:", rob2(nums))

方法3:优化的动态规划

计算过程:

prev = 2, curr = 7
prev = 7, curr = 9
prev = 9, curr = 10
prev = 10, curr = 12
结果:12

代码如下:

python">def rob3(nums):if not nums:return 0if len(nums) == 1:return nums[0]prev, curr = 0, 0for num in nums:prev, curr = curr, max(prev + num, curr)return curr# 测试
nums = [2, 7, 9, 3, 1]
print("最高金额:", rob3(nums))

方法4:递归

计算过程:

helper(0) = max(helper(1), helper(2) + 2) = max(7, 9) = 9
helper(1) = max(helper(2), helper(3) + 7) = max(9, 10) = 10
helper(2) = max(helper(3), helper(4) + 9) = max(10, 12) = 12
helper(3) = max(helper(4), helper(5) + 3) = max(9, 12) = 12
helper(4) = max(helper(5), 1) = 12
结果:12

代码如下:

python">def rob4(nums):def helper(i):if i == len(nums):return 0if i == len(nums) - 1:return nums[i]return max(helper(i+1), nums[i] + helper(i+2))return helper(0)# 测试
nums = [2, 7, 9, 3, 1]
print("最高金额:", rob4(nums))

总结

这个问题在算法学习中非常重要,因为它展示了如何使用动态规划解决具有重叠子问题和最优子结构特性的问题。它也常用于面试中,考察候选人对动态规划的理解和应用能力。

这个问题的变种也很多,比如考虑环形街道的情况,或者房屋之间的防盗系统有不同的触发条件等。


http://www.ppmy.cn/embedded/127412.html

相关文章

MySQL-07.DDL-表结构操作-数据类型

一.MySQL中的数据类型 MySQL中的数据类型主要分为3种:数字类型,字符串类型,日期时间类型。如下图所示! 二.数值类型 三.字符串类型 四.日期类型

深度解析python标准库模块json库!

在 Python 中,json库用于处理 JSON(JavaScript Object Notation)数据。以下是json库的主要用法: 一、将 Python 对象转换为 JSON 字符串 使用dumps()方法: json.dumps()方法可以将 Python 对象转换为 JSON 字符串。例…

高校新生报道管理系统使用SpringBootSSM框架开发

!!!页面底部,文章结尾,加我好友,获取计算机毕设开发资料 目录 一、引言 二、相关技术介绍 三、系统需求分析 四、系统设计 五、关键技术实现 六、测试与优化 七、总结与展望 一、引言 当前高校新生报到过程中存在许多问题,…

spring02 IOC DI

IOC、DI 1. Spring 给对象的属性赋值的两种方式 1. 有参构造注入 1. 必须提供有参构造方法 AllArgsConstructor public class Student { } 2. applicationContext.xml配置 <!--通过有参构造方法给属性赋值&#xff0c;创建Student组件对象--><bean id"stu&quo…

高校学科竞赛平台开发:SpringBoot技术选型与应用

3系统分析 3.1可行性分析 通过对本高校学科竞赛平台实行的目的初步调查和分析&#xff0c;提出可行性方案并对其一一进行论证。我们在这里主要从技术可行性、经济可行性、操作可行性等方面进行分析。 3.1.1技术可行性 本高校学科竞赛平台采用SSM框架&#xff0c;JAVA作为开发语…

Kafka之资源容量评估

编写目的意义 应用场景为如果有租户需要部署kafka集群&#xff0c;并给出业务压力&#xff0c;根据业务评估kafka资源情况&#xff0c;如cpu 磁盘 内存 带宽等维度。为业务解决因资源过小故障和新业务部署提供了参考和计算方法&#xff0c;减少后续的维护成本 资源容量评估 …

树莓派应用--AI项目实战篇来啦-15.SSD Mobilenet V3目标检测

1. Mobilenet 介绍 Mobilenet 是一种专为移动和嵌入式视觉应用而设计的卷积神经网络。它们不使用标准的卷积层&#xff0c;而是基于使用深度可分离卷积的简化架构&#xff0c;使用这种架构&#xff0c;我们可以为移动和嵌入式设备&#xff08;例如&#xff1a;树莓派&#xff0…

MySQL 之索引和查询优化

在 MySQL 数据库中&#xff0c;索引是提高查询性能的重要手段之一。而理解和应用最左前缀原则对于有效地利用索引进行查询优化至关重要。 一、索引的作用 索引是一种数据结构&#xff0c;它可以帮助数据库系统快速地定位和检索数据。通过在表的某些列上创建索引&#xff0c;数…