动态规划学习——多状态dp(打家劫舍问题)

一,打家劫舍I

题目:

一个专业的小偷,计划偷窃沿街的房屋。每间房内都藏有一定的现金,影响小偷偷窃的唯一制约因素就是相邻的房屋装有相互连通的防盗系统,如果两间相邻的房屋在同一晚上被小偷闯入,系统会自动报警

给定一个代表每个房屋存放金额的非负整数数组 nums ,请计算不触动警报装置的情况下,一夜之内能够偷窃到的最高金额。

题目接口:

cpp 复制代码
class Solution {
public:
    int rob(vector<int>& nums) {

    }
};

解题思路及代码:

这个打家劫舍问题便是经典的多状态dp问题。为什么是多状态dp问题呢?

**1.**首先这道题要我们求得是最大值,并且第i家的最大值其实是由前面的前i-1家推导出来的,所以这个问题可以是一个dp问题。

**2.**为什么1是个多状态dp问题呢?首先我们知道,我们第i天的最大值是由前面的i-1家推导出来的。但是前面的i-1家里面的每一天其实都有偷与不偷两种状态。所以综上两点我们的这道题目便是一个多状态dp问题了。

解决这种dp问题的第一步便是画出状态转移的图表示,如下:

首先解释一下箭头,箭头的开始表示的是前一家的状态,箭头的结束表示这一家的状态。

所以要变成这一家不偷的状态便有两种情况:1.上一家偷了,这一家我不偷

2.上一家没偷,这一家我还是不偷。

今天变成偷的状态:上一家没偷,这一家偷。(上一家偷了,这一家就不能偷了。题目要求)

有了这一个状态转移关系以后,我们便可以开始我们的题目解答了,代码如下:

cpp 复制代码
class Solution {
public:
    int rob(vector<int>& nums) {
        int n = nums.size();

        //定义两个数组分别代表第i家偷了和没偷的最大值,f表示偷了,g表式没偷
        vector<int>f(n);
        auto g = f;

        //初始化
         f[0] = nums[0];//第一家偷了
         g[0] = 0;//第一家没偷
         
         //根据状态转移图填表
         for(int i = 1;i<n;i++)
         {
             f[i] = g[i-1]+nums[i];//上一家没偷,这一家才能偷
             g[i] = max(g[i-1],f[i-1]);//不管上一家偷不偷,这一家都不偷,取前面一家在两种状态下的最大值。
         }

         return max(g[n-1],f[n-1]);//返回每一家都到访后的最大值
    }
};
相关推荐
_深海凉_1 天前
LeetCode热题100-路径总和 III
算法·leetcode·职场和发展
RTC老炮1 天前
WebRTC AEC3 算法原理分析
算法·webrtc
炽烈小老头1 天前
【每天学习一点算法 2026/05/20】省份数量
学习·算法
乐迪信息1 天前
乐迪信息:港口夜间船舶巡查难,AI摄像机法全天候监测
人工智能·物联网·算法·计算机视觉·目标跟踪
sali-tec1 天前
C# 基于OpenCv的视觉工作流-章74-线-线距离
图像处理·人工智能·opencv·算法·计算机视觉
故事和你911 天前
洛谷-【图论2-3】最小生成树1
开发语言·数据结构·c++·算法·动态规划·图论
故事和你911 天前
洛谷-【图论2-3】最小生成树2
开发语言·数据结构·c++·算法·动态规划·图论
清平乐的技术专栏1 天前
【Flink学习】(七)Flink 状态编程入门,有状态实时计算
大数据·学习·flink
guygg881 天前
贝叶斯非局部均值降噪算法C语言实现
c语言·算法·均值算法
郝学胜-神的一滴1 天前
中级OpenGL教程 006:高光反射原理与 Shader 实现
c++·unity·godot·图形渲染·three.js·opengl·unreal