打家劫舍问题的动态规划解法与性能优化笔记

打家劫舍问题的动态规划解法与性能优化笔记

一、问题背景回顾

给定一个非负整数数组 nums,每个元素代表对应房屋存放的金额,要求在不偷窃相邻房屋(避免触发警报)的前提下,求解能偷窃到的最大金额。这一问题的核心是在约束条件下寻找最优解,具备动态规划问题典型的"最优子结构"特征------当前位置的最优解可由前序子问题的解推导而来。

二、基础解法:常规动态规划思路

2.1 状态定义与转移

首先从最直观的动态规划思路入手,定义 dp[i] 表示前 i 间房屋能偷窃到的最大金额。对于第 i 间房屋,存在两种选择:

  • 偷第 i 间:则第 i-1 间不能偷,此时 dp[i] = dp[i-2] + nums[i]
  • 不偷第 i 间:此时 dp[i] = dp[i-1]

因此状态转移方程为:dp[i] = max(dp[i-1], dp[i-2] + nums[i])

2.2 基础实现

cpp 复制代码
#include <vector>
#include <algorithm>
using namespace std;

class Solution {
public:
    int rob(vector<int>& nums) {
        int n = nums.size();
        if (n == 0) return 0;
        if (n == 1) return nums[0];
        
        // 定义dp数组存储前i间房屋的最大金额
        vector<int> dp(n);
        dp[0] = nums[0];
        dp[1] = max(nums[0], nums[1]);
        
        for (int i = 2; i < n; i++) {
            dp[i] = max(dp[i-1], dp[i-2] + nums[i]);
        }
        return dp[n-1];
    }
};

2.3 基础解法分析

  • 时间复杂度:O(n),需遍历数组一次完成状态转移;
  • 空间复杂度:O(n),需维护一个长度为 ndp 数组存储中间状态。

这一解法逻辑清晰,符合动态规划的常规思路,能正确解决问题,但在数据规模较大时,数组的空间开销会成为可优化的点。

三、空间优化:压缩状态存储

3.1 优化思路

观察状态转移方程可发现,计算 dp[i] 仅依赖 dp[i-1]dp[i-2] 两个值,无需保存完整的 dp 数组。因此可使用两个变量替代数组,分别记录前两步的状态:

  • first:对应 dp[i-2],即前 i-2 间房屋的最大金额;
  • second:对应 dp[i-1],即前 i-1 间房屋的最大金额。

遍历过程中只需不断更新这两个变量,即可推导出当前的最优解,无需额外存储所有中间状态。

3.2 优化后实现

cpp 复制代码
#include <vector>
#include <algorithm>
using namespace std;

class Solution {
public:
    int rob(vector<int>& nums) {
        int n = nums.size();
        if (n == 1) {
            return nums[0];
        }
        // 初始化前两步状态
        int first = nums[0];
        int second = max(nums[0], nums[1]);
        int result = second;
        
        for (int i = 2; i < n; i++) {
            // 状态转移:偷或不偷当前房屋,取最大值
            result = max(first + nums[i], second);
            // 更新状态,为下一次遍历做准备
            first = second;
            second = result;
        }
        return result;
    }
};

3.3 优化后分析

  • 时间复杂度:仍为 O(n),遍历次数未发生变化;
  • 空间复杂度:优化为 O(1),仅使用有限个变量,空间开销与输入规模无关。

这一优化是动态规划问题中"状态压缩"的典型应用,在仅依赖前有限步状态的场景中,能显著降低空间占用,且不会增加时间成本。

四、工程实现的细节考量

4.1 边界条件处理

代码中优先处理 n == 1 的情况,避免后续访问 nums[1] 时出现数组越界。在工程实现中,边界条件的处理是保证代码鲁棒性的关键------实际场景中输入规模可能存在极端情况(如空数组、单元素数组),需提前预判并规避异常。

4.2 变量初始化的合理性

初始时 secondmax(nums[0], nums[1]),符合"前两间房屋只能选金额更高者"的逻辑,这一初始化方式既贴合问题规则,也为后续遍历奠定了正确的初始状态。在工程开发中,变量初始化的合理性直接影响后续逻辑的正确性,需与问题的实际约束一致。

4.3 代码可读性与可维护性

优化后的代码未因追求性能而牺牲可读性:变量命名(first/second)直观反映其对应的状态含义,核心逻辑(状态转移、变量更新)分步骤实现,便于后续调试和扩展。例如,若问题扩展为"房屋环形排列"(首尾房屋也不能同时偷),仅需在现有逻辑基础上稍作调整,即可适配新场景。

五、进一步的思考

5.1 时间复杂度的上限

该问题的时间复杂度已达到 O(n),这是理论上的最优值------因为要确定每间房屋的选择策略,必须遍历所有房屋至少一次,无法通过算法优化进一步降低时间复杂度。

5.2 状态压缩的适用场景

状态压缩并非适用于所有动态规划问题,其核心前提是"当前状态仅依赖有限的前序状态"。例如,若问题约束变为"不能偷相邻的三间房屋",则需保存前三个状态,但仍可通过变量替代数组实现空间优化;而若状态依赖的前序步骤数与输入规模成正比,则状态压缩无实际意义。

5.3 实际应用中的权衡

在工程实践中,空间优化的优先级需结合实际场景判断:

  • 若输入规模较小(如房屋数量少于1000),基础解法的数组开销可忽略,此时优先保证代码可读性;
  • 若输入规模极大(如百万级房屋数据),空间优化能显著降低内存占用,避免内存溢出,此时状态压缩是必要选择。

总结

  1. 打家劫舍问题的核心是利用动态规划的最优子结构特性,通过前序子问题的解推导当前最优解,基础解法通过 dp 数组实现,空间复杂度为 O(n)
  2. 利用"状态仅依赖前两步"的特征,可通过两个变量替代 dp 数组,将空间复杂度优化至 O(1),且不影响时间效率;
  3. 工程实现中需关注边界条件、变量初始化等细节,同时结合实际场景权衡性能优化与代码可读性的关系,保证代码的鲁棒性和可维护性。
相关推荐
二哈赛车手12 小时前
新人笔记---ApiFox的一些常见使用出错
java·笔记·spring
吃好睡好便好13 小时前
在Matlab中绘制横直方图
开发语言·学习·算法·matlab
仰泳之鹅13 小时前
【C语言】自定义数据类型2——联合体与枚举
c语言·开发语言·算法
xian_wwq15 小时前
【学习笔记】AGC协调控制系统概述
笔记·学习
x_yeyue15 小时前
三角形数
笔记·算法·数论·组合数学
憧憬成为java架构高手的小白16 小时前
docker学习笔记(基于b站多个视频学习)【未完结】
笔记·学习
念何架构之路16 小时前
Go语言加密算法
数据结构·算法·哈希算法
AI科技星16 小时前
《数学公理体系·第三部·数术几何》(2026 年版)
c语言·开发语言·线性代数·算法·矩阵·量子计算·agi
失去的青春---夕阳下的奔跑17 小时前
560. 和为 K 的子数组
数据结构·算法·leetcode
黎阳之光17 小时前
黎阳之光:以视频孪生重构智慧医院信息化,打造高标项目核心竞争力
大数据·人工智能·物联网·算法·数字孪生