【Leetcode每日一题】 动态规划 - 简单多状态 dp 问题 - 打家劫舍 II(难度⭐⭐)(67)

1. 题目解析

题目链接:213. 打家劫舍 II

这个问题的理解其实相当简单,只需看一下示例,基本就能明白其含义了。

2.算法原理

这个问题是经典的"打家劫舍"问题的变种,原问题是在单排房屋中进行偷窃,而这个问题则是在环形排列的房屋中进行。环形排列的特点在于首尾相连,这为我们设计算法带来了新的挑战。然而,通过一些巧妙的转换,我们可以将这个问题分解为两个单排问题来解决。

首先,我们需要明确环形排列带来的限制:由于首尾相连,我们不能同时偷取第一个和最后一个房屋,因为这会导致连续的偷窃行为被发现。因此,我们可以将问题拆分为两种情况来考虑:

  1. 偷取第一个房屋时的最大金额(设为x):在这种情况下,我们不能偷取最后一个房屋,因为这将构成一个闭环。因此,我们的搜索范围变成了从第一个房屋到倒数第二个房屋,即区间[0, n-2]。

  2. 不偷取第一个房屋时的最大金额(设为y):在这种情况下,我们可以偷取最后一个房屋,因为第一个房屋已经被排除在外,不会构成闭环。因此,我们的搜索范围变成了从第二个房屋到最后一个房屋,即区间[1, n-1]。

通过分别计算这两种情况下的最大金额,我们可以得到两个结果:x和y。最终,我们只需要取这两个结果中的较大值,即为在环形排列的房屋中进行偷窃能够获得的最大金额。

3.代码编写

cpp 复制代码
class Solution {
public:
    int rob(vector<int>& nums) {
        int n = nums.size();
        return max(nums[0] + rob1(nums, 2, n - 2), rob1(nums, 1, n - 1));
    }
    int rob1(vector<int>& nums, int left, int right) {
        if(left > right) return 0;
        int n = nums.size();
        vector<int> f(n), g(n);
        f[left] = nums[left]; // 初始化
        for (int i = left + 1; i <= right; i++) {
            f[i] = g[i - 1] + nums[i];
            g[i] = max(f[i - 1], g[i - 1]);
        }
        return max(f[right], g[right]);
    }
};

The Last

嗯,就是这样啦,文章到这里就结束啦,真心感谢你花时间来读。

觉得有点收获的话,不妨给我点个吧!

如果发现文章有啥漏洞或错误的地方,欢迎私信我或者在评论里提醒一声~

相关推荐
源代码•宸4 小时前
分布式缓存-GO(分布式算法之一致性哈希、缓存对外服务化)
开发语言·经验分享·分布式·后端·算法·缓存·golang
旖旎夜光5 小时前
多态(11)(下)
c++·学习
yongui478345 小时前
MATLAB的指纹识别系统实现
算法
高山上有一只小老虎5 小时前
翻之矩阵中的行
java·算法
yangpipi-5 小时前
《C++并发编程实战》 第4章 并发操作的同步
开发语言·c++
jghhh015 小时前
RINEX文件进行卫星导航解算
算法
Chance_to_win5 小时前
C++基础知识
c++
爱思德学术5 小时前
中国计算机学会(CCF)推荐学术会议-A(计算机科学理论):LICS 2026
算法·计算机理论·计算机逻辑
CVHub5 小时前
多模态图文训推一体化平台 X-AnyLabeling 3.0 版本正式发布!首次支持远程模型推理服务,并新增 Qwen3-VL 等多款主流模型及诸多功能特性,等
算法
有趣的我5 小时前
C++ 多态介绍
开发语言·c++