算法训练之动态规划(五)——简单多状态问题

news2025/4/12 17:07:18


♥♥♥~~~~~~欢迎光临知星小度博客空间~~~~~~♥♥♥

♥♥♥零星地变得优秀~也能拼凑出星河~♥♥♥

♥♥♥我们一起努力成为更好的自己~♥♥♥

♥♥♥如果这一篇博客对你有帮助~别忘了点赞分享哦~♥♥♥

♥♥♥如果有什么问题可以评论区留言或者私信我哦~♥♥♥

✨✨✨✨✨✨ 个人主页✨✨✨✨✨✨

        这一篇博客我们继续来看看动态规划系列里面的简单多状态问题,准备好了吗~我们发车去探索奥秘啦~🚗🚗🚗🚗🚗🚗

目录

粉刷房子

买卖股票的最佳时机(含冷冻期)


粉刷房子

粉刷房子

        可以看到题目要求给房子上颜色,并且相邻的房子颜色不能相同~这显然是是一个多状态的问题,接下来我们来一步步分析一下~

分析:

1、状态表示

        题目要求:相邻的房子颜色不能相同,每一个房子有三种颜色可以选择~我们创建三个dp表来进行表示,事实上,题目给出的二维数组,行号就是房子号数,列号是涂某一种颜色的花费,我们也可以用这样一个二维数组来形成我们的dp表~每一个房子都有三种不同的情况~

        结合这里的题目要求+经验:

        dp表中的dp1[i][0]表示为到达该位置并且选择该位置选择涂红色的最小花费~

        dp表中的dp1[i][1]表示为到达该位置并且选择该位置选择涂蓝色的最小花费~

        dp表中的dp1[i][2]表示为到达该位置并且选择该位置选择涂绿色的最小花费~

2、状态转移方程

       我们以离【i】位置最近的状态分析状态转移方程,处理dp表

1、

        dp【i】【0】选择【i】位置涂红色,那么说明前面的位置是一定不可以涂蓝色和绿色的,取两者最小值再加上当前位置涂红色的花费,那么

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

2、

        dp【i】【1】选择【i】位置涂蓝色,那么说明前面的位置是一定不可以涂红色和绿色的,取两者最小值再加上当前位置涂蓝色的花费,那么

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

3、

        dp【i】【2】选择【i】位置涂绿色,那么说明前面的位置是一定不可以涂蓝色和红色的,取两者最小值再加上当前位置涂绿色的花费,那么

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

3、初始化

        我们可以看到,状态转移方程里面有i-1当i=0的时候显然会出现越界的情况,所以我们需要进行初始化

        结合前面如果不想初始化太麻烦,我们可以多申请一些空间,但是事实上这个题目初始化比较简单,直接初始化dp[0][0],dp[0][1],dp[0][2]就可以了,所以我们直接进行初始化~

        dp[0][0]就是选择0位置涂红色花费nums[0][0], dp[0][1]就是选择0位置涂蓝色花费nums[0][1], dp[0][2]就是选择0位置涂绿色花费nums[0][2]那么我们初始化结果就是

                dp[0][0]=nums[0][0] , dp[0][1]=nums[0][1] , dp[0][2]=nums[0][2]

4、填表顺序

        我们这里的逻辑是从前面依次推出后面的,所以填表顺序是从前往后

5、返回结果

      这里返回结果是到最后一个房子的最小花费,最后一个房子有三种情况,一种是选择涂红色,一种是选择涂蓝色,还有一种是选择涂绿色,返回三种情况最小值就可以了,即返回min(min(dp[m-1][0],dp[m-1][1]),dp[m-1][2])

注意点:结合题目给出的范围,这里不需要处理边界情况~

代码实现:

class Solution
{
public:
    int minCost(vector<vector<int>>& costs)
    {
        //1、创建dp表
        int m = costs.size();//房子号
        int n = costs[0].size();//颜色
        vector<vector<int>> dp(m, vector<int>(n));
        //2、初始化
        dp[0][0] = costs[0][0];
        dp[0][1] = costs[0][1];
        dp[0][2] = costs[0][2];
        //3、填表
        for (int i = 1; i < m; i++)
        {
            dp[i][0] = min(dp[i - 1][1], dp[i - 1][2]) + costs[i][0];
            dp[i][1] = min(dp[i - 1][0], dp[i - 1][2]) + costs[i][1];
            dp[i][2] = min(dp[i - 1][0], dp[i - 1][1]) + costs[i][2];
        }
        //4、返回结果
        return min(min(dp[m - 1][0], dp[m - 1][1]), dp[m - 1][2]);
    }
};

顺利通过~

我们也可以采用有趣提到的增加虚拟节点的方法,这里相当于多增加一个空房子,有下面两个注意点:

1、虚拟节点值是多少?

        虚拟节点会影响到第一个房子的最小花费,事实上,到第一个房子的最小花费也就是第一个房子的花费,不想让虚拟节点影响,就把虚拟节点值设置为0~

2、注意下标映射关系

        相当于增加了一个房子,注意与原来的数组的下标映射~

代码实现:

class Solution
{
public:
    int minCost(vector<vector<int>>& costs)
    {
        //使用虚拟节点
        //1、创建dp表
        int m = costs.size();
        int n = costs[0].size();//也可以直接写为已知的3
        vector<vector<int>> dp(m + 1, vector<int>(n));
        //2、初始化虚拟节点
        dp[0][0] = 0;
        dp[0][1] = 0;
        dp[0][2] = 0;
        //3、填表
        for (int i = 1; i <= m; i++)
        {
            //注意下标映射关系
            dp[i][0] = min(dp[i - 1][1], dp[i - 1][2]) + costs[i - 1][0];
            dp[i][1] = min(dp[i - 1][0], dp[i - 1][2]) + costs[i - 1][1];
            dp[i][2] = min(dp[i - 1][0], dp[i - 1][1]) + costs[i - 1][2];
        }
        //4、返回结果
        return min(min(dp[m][0], dp[m][1]), dp[m][2]);
    }
};

顺利通过~不难发现,代码量其实是差不多的,大家选择自己喜欢的方式就好~

买卖股票的最佳时机(含冷冻期)

买卖股票的最佳时机(含冷冻期)

        这个题目显然是一个多状态问题,那么我们首先得分析它有哪几个状态:

1、买入状态(也就是手上有股票的状态,可以进行卖出)

2、冷冻期状态(不可以进行买入)

3、可以交易的状态(可以进行买入)

接下来画图分析这几个状态之间的关系(也就是讨论状态相互之间是否可达以及是否可以自己到自己)

知道了这三个状态之间的关系,我们就可以利用动态规划的思想进行分析:

1、状态表示

        题目要求:既然有三个状态,那么我们就需要创建三个dp表表示不同位置可能的状态~

        结合这里的题目要求+经验:

        dp1表中的dp1[i]表示为到达该位置进行操作后处于买入状态的最大利润~

        dp2表中的dp2[i]表示为到达该位置进行操作后处于可交易状态的最大利润~

        dp3表中的dp3[i]表示为到达该位置进行操作后处于冷冻期状态的最大利润~

        注意是在该位置进行操作后处于什么状态,而不是到达该位置是什么状态,这样会比较麻烦~

2、状态转移方程

       我们以离【i】位置最近的状态分析状态转移方程,处理dp表

1、

       怎么样会处于买入状态呢?结合前面的画图分析可能是前一天可交易状态下,在今天买入股票变成买入状态;也可能是前一天买入状态下,今天什么都不干依然是买入状态,取两种情况的较大值~

        dp1表状态转移方程:

        dp1[i]=max(dp1[i-1],dp2[i-1]-prices[i]);

2、

        怎么样会处于可交易状态呢?结合前面的画图分析可能是前一天冷冻期状态,今天就是可交易的状态;也可能是前一天可交易状态下,今天什么都不干依然是可交易状态,取两种情况的较大值~

        dp2表状态转移方程:

        dp2[i]=max(dp2[i-1],dp3[i-1]);

3、

       怎么样会处于冷冻期状态呢?结合前面的画图分析可能是前一天处于买入状态,在今天进行卖出也就处于冷冻期状态了~没有其他情况

        dp3表状态转移方程:

        dp3[i]=dp1[i-1]+prices[i];

3、初始化

        我们可以看到,状态转移方程里面有i-1当i=0的时候显然会出现越界的情况,所以我们需要进行初始化

        结合前面如果不想初始化太麻烦,我们可以多申请一些空间,但是事实上这个题目初始化比较简单,直接初始化dp1[0],dp2[0],dp3[0]就可以了,所以我们直接进行初始化~

        dp1[0]就是第一天操作后处于买入状态,那么利润为-prices[0];

        dp2[0]就是第一天操作后处于可交易状态,那就是什么都不干,那么利润为0;

        dp3[0]就是第一天操作后处于冷冻期状态,这是不可能的,那么利润为0;

        那么我们初始化结果就是

                dp1[0]=-prices[0] , dp2[0]=0 , dp3[0]=0

4、填表顺序

        我们这里的逻辑是从前面依次推出后面的,所以填表顺序是从前往后

5、返回结果

      这里返回结果是到最后一天的最大利润,最后一天有三种情况,返回三种情况最大值就可以了,即返回return max(max(dp1[n-1],dp2[n-1]),dp3[n-1]);

        当然,最后一天是不可能还处于买入状态的,这样就亏了,也就可以返回return max(dp2[n-1],dp3[n-1]);

注意点:结合题目给出的范围,这里不需要处理边界情况~

代码实现:

class Solution 
{
public:
    int maxProfit(vector<int>& prices) 
    {
        //1、创建dp表
        int n=prices.size();
        vector<int> dp1(n);//买入状态
        vector<int> dp2(n);//可以交易状态
        vector<int> dp3(n);//冷冻期状态

        //2、初始化
        dp1[0]=-prices[0];
        dp2[0]=0;
        dp3[0]=0;

        //3、填表
        for(int i=1;i<n;i++)
        {
            dp1[i]=max(dp1[i-1],dp2[i-1]-prices[i]);
            dp2[i]=max(dp2[i-1],dp3[i-1]);
            dp3[i]=dp1[i-1]+prices[i];
        }

        //4、返回结果
        //return max(max(dp1[n-1],dp2[n-1]),dp3[n-1]);
        return max(dp2[n-1],dp3[n-1]);
    }
};

顺利通过~

除了这种创建dp表的方式,我们也可以像前面那样创建二维数组(n*3)来实现三个dp表~

我们重新来进行分析一下:

1、状态表示

        题目要求:既然有三个状态,那么我们就需要创建三个dp表表示不同位置可能的状态~这里创建一个n*3的二维数组来表示~

        结合这里的题目要求+经验:

        dp表中的dp[i][0]表示为到达该位置进行操作后处于买入状态的最大利润~

        dp表中的dp[i][1]表示为到达该位置进行操作后处于可交易状态的最大利润~

        dp表中的dp[i][2]表示为到达该位置进行操作后处于冷冻期状态的最大利润~

2、状态转移方程

       我们以离【i】位置最近的状态分析状态转移方程,处理dp表

1、

       怎么样会处于买入状态呢?结合前面的画图分析可能是前一天可交易状态下,在今天买入股票变成买入状态;也可能是前一天买入状态下,今天什么都不干依然是买入状态,取两种情况的较大值~

        dp表状态转移方程:

        dp[i][0]=max(dp[i-1][0],dp[i-1][1]-prices[i]);

2、

        怎么样会处于可交易状态呢?结合前面的画图分析可能是前一天冷冻期状态,今天就是可交易的状态;也可能是前一天可交易状态下,今天什么都不干依然是可交易状态,取两种情况的较大值~

        dp表状态转移方程:

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

3、

       怎么样会处于冷冻期状态呢?结合前面的画图分析可能是前一天处于买入状态,在今天进行卖出也就处于冷冻期状态了~没有其他情况

        dp表状态转移方程:

        dp[i][2]=dp[i-1][0]+prices[i];

3、初始化

        我们可以看到,状态转移方程里面有i-1当i=0的时候显然会出现越界的情况,所以我们需要进行初始化

        结合前面如果不想初始化太麻烦,我们可以多申请一些空间,但是事实上这个题目初始化比较简单,直接初始化dp[0][0],dp[0][1],dp[0][2]就可以了,所以我们直接进行初始化~

        dp[0][0]就是第一天操作后处于买入状态,那么利润为-prices[0];

        dp[0][1]就是第一天操作后处于可交易状态,那就是什么都不干,那么利润为0;

        dp[0][2]就是第一天操作后处于冷冻期状态,这是不可能的,那么利润为0;

        那么我们初始化结果就是

               

        dp[0][0]=-prices[0];//买入状态

        dp[0][1]=0;//可交易状态

        dp[0][2]=0;//冷冻期状态

4、填表顺序

        我们这里的逻辑是从前面依次推出后面的,所以填表顺序是从前往后

5、返回结果

      这里返回结果是到最后一天的最大利润,最后一天有三种情况,返回三种情况最大值就可以了,即返回return max(max(dp[n-1][0],dp[n-1][1]),dp[n-1][2]);

        当然,最后一天是不可能还处于买入状态的,这样就亏了,也就可以返回return max(dp[n-1][1],dp[n-1][2]);

代码实现:(逻辑都是差不多的,只不过实现上有区别)


class Solution
{
public:
    int maxProfit(vector<int>& prices)
    {
        //1、创建二维数组dp表
        int n = prices.size();
        vector<vector<int>> dp(n, vector<int>(3, 0));

        //2、初始化
        dp[0][0] = -prices[0];//买入状态
        dp[0][1] = 0;//可交易状态
        dp[0][2] = 0;//冷冻期状态

        //3、填表
        for (int i = 1; i < n; i++)
        {
            dp[i][0] = max(dp[i - 1][0], dp[i - 1][1] - prices[i]);
            dp[i][1] = max(dp[i - 1][1], dp[i - 1][2]);
            dp[i][2] = dp[i - 1][0] + prices[i];
        }

        //4、返回结果
        return max(max(dp[n - 1][0], dp[n - 1][1]), dp[n - 1][2]);
        //return max(dp[n-1][1],dp[n-1][2]);
    }
};

顺利通过~


♥♥♥本篇博客内容结束,期待与各位优秀程序员交流,有什么问题请私信♥♥♥

♥♥♥如果这一篇博客对你有帮助~别忘了点赞分享哦~♥♥♥

✨✨✨✨✨✨个人主页✨✨✨✨✨✨


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

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

相关文章

SVMSPro分布式综合安防管理平台-->以S3存储革新,开启智能安防新纪元

SVMSPro分布式综合安防管理平台–>以S3存储革新&#xff0c;开启智能安防新纪元 在数字化转型浪潮下&#xff0c;企业安防管理正面临海量数据存储、跨区域协同以及数据安全的严峻挑战。如何实现高效、弹性、低成本的存储扩容&#xff1f;如何确保关键录像数据万无一失&…

脑科学与人工智能的交叉:未来智能科技的前沿与机遇

引言 随着科技的迅猛发展&#xff0c;脑科学与人工智能&#xff08;AI&#xff09;这两个看似独立的领域正在发生深刻的交汇。脑机接口、神经网络模型、智能机器人等前沿技术&#xff0c;正带来一场跨学科的革命。这种结合不仅推动了科技进步&#xff0c;也在医疗、教育、娱乐等…

docker 运行自定义化的服务-后端

docker 运行自定义化的服务-前端-CSDN博客 运行自定义化的后端服务 具体如下&#xff1a; ①打包后端项目&#xff0c;形成jar包 ②编写dockerfile文件&#xff0c;文件内容如下&#xff1a; # 使用官方 OpenJDK 镜像 FROM jdk8:1.8LABEL maintainer"ATB" version&…

NO.82十六届蓝桥杯备战|动态规划-从记忆化搜索到动态规划|下楼梯|数字三角形(C++)

记忆化搜索 在搜索的过程中&#xff0c;如果搜索树中有很多重复的结点&#xff0c;此时可以通过⼀个"备忘录"&#xff0c;记录第⼀次搜索到的结果。当下⼀次搜索到这个结点时&#xff0c;直接在"备忘录"⾥⾯找结果。其中&#xff0c;搜索树中的⼀个⼀个结点…

【时时三省】(C语言基础)用switch语句实现多分支选择结构 例题

山不在高&#xff0c;有仙则名。水不在深&#xff0c;有龙则灵。 ----CSDN 时时三省 例题&#xff1a; 用switch语句处理菜单命令。在许多应用程序中&#xff0c;用菜单对流程进行控制&#xff0c;例如从键盘输入一个 A 或 a 字符&#xff0c;就会执行A操作&#xff0c;输入一…

全域数字化:从“智慧城市”到“数字生命体”的进化之路

一、国家战略下的城市数字化浪潮 2024年5月&#xff0c;国家四部委联合发布《关于深化智慧城市发展 推进城市全域数字化转型的指导意见》&#xff0c;明确提出以数据为引擎&#xff0c;系统性重塑城市技术架构与管理流程&#xff0c;推动城市治理迈向“全域协同、数实融合”的…

基于SSM的线上花店鲜花销售商城网站系统

作者&#xff1a;计算机学姐 开发技术&#xff1a;SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等&#xff0c;“文末源码”。 专栏推荐&#xff1a;前后端分离项目源码、SpringBoot项目源码、Vue项目源码、SSM项目源码、微信小程序源码 精品专栏&#xff1a;…

DAPP实战篇:使用web3.js连接合约

说明 本系列内容目录:专栏:区块链入门到放弃查看目录 如果你还没有创建好项目请先查看:《DApp实战篇:先用前端起个项目》,如果你还不知道web3.js是什么请先查看:《DApp实战篇:前端技术栈一览》。 安装 点此查看web3.js官方文档 打开项目根目录,并唤起终端: 键入w…

K8S-证书过期更新

K8S证书过期问题 K8S证书过期处理方法 Unable to connect to the server: x509: certificate has expired or is not yet valid 1、查看证书有效期&#xff1a; # kubeadm certs check-expiration2、备份证书 # cp -rp /etc/kubernetes /etc/kubernetes.bak3、直接重建证书 …

蓝桥杯第十五届真题——握手问题

#include<bits/stdc.h> using namespace std; int main() {ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);int sum0;for(int i7;i<49;i){sumi;}cout<<sum;return 0; }

5G_WiFi_CE_DFS

目录 一、规范要求 1、法规目录 2、定义 3、运行模式 4、主/从设备相关的运行行为及具体的动态频率选择&#xff08;DFS&#xff09;要求 5、产品角色确定测试项目 6、测试项目 测试项1&#xff1a;信道可用性检查&#xff08;Channel Availability Check&#xff09; …

springboot 处理编码的格式为opus的音频数据解决方案【java8】

opus编码的格式概念&#xff1a; Opus是一个有损声音编码的格式&#xff0c;由Xiph.Org基金会开发&#xff0c;之后由IETF&#xff08;互联网工程任务组&#xff09;进行标准化&#xff0c;目标是希望用单一格式包含声音和语音&#xff0c;取代Speex和Vorbis&#xff0c;且适用…

RK3568 基于Gstreamer的多媒体调试记录

文章目录 1、环境介绍2、概念理清3、提前准备4、GStreamer编译5、GStreamer基础介绍6、视频播放初体验7、视频硬编码7.1、h2647.2、h265 8、视频硬解码8.1、解码视频并播放8.2、解码视频并播放带音频 1、环境介绍 硬件&#xff1a;飞凌ok3568-c开发板 软件&#xff1a;原厂rk…

VS Code 的 .S 汇编文件里面的注释不显示绿色

1. 确认文件语言模式 打开 .S 文件后&#xff0c;查看 VS Code 右下角的状态栏&#xff0c;确认当前文件的识别模式&#xff08;如 Assembly、Plain Text 等&#xff09;。如果显示为 Plain Text 或其他非汇编模式&#xff1a; 点击状态栏中的语言模式&#xff08;如 Plain Te…

5分钟读懂ArgoCD:在Kubernetes中实现持续部署

Kubernetes中的Argo CD介绍 Argo CD是用于Kubernetes的声明式GitOps持续交付工具。它遵循GitOps模式&#xff0c;以Git仓库作为定义所需应用程序状态的唯一真实来源&#xff0c;能在指定的目标环境中自动部署应用程序&#xff0c;并持续监控应用程序的运行状态&#xff0c;确保…

cs224w课程学习笔记-第10课

cs224w课程学习笔记-第10课 异构图 前言一、异构图1、异构图定义2、异构图与同构图 二、异构图下的GNN1、GCN扩展至RGCN1.1 RGCN原理1.2 异构图的任务预测特点1.3 异构图任务预测基础案例 2、完整的异构图GCN三、异构图下的Transformer 前言 异构图的定义是节点内部存在类型不…

OpenCV 图形API(26)图像滤波-----方框滤波函数boxFilter()

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 使用方框滤波器模糊图像。 该函数使用以下内核来平滑图像&#xff1a; K α [ 1 1 … 1 1 1 … 1 ⋮ ⋮ ⋱ ⋮ 1 1 … 1 ] K \alpha \begin{b…

大模型上下文协议MCP详解(2)—核心功能

版权声明 本文原创作者:谷哥的小弟作者博客地址:http://blog.csdn.net/lfdfhl1. 标准化上下文交互技术 1.1 实时数据接入能力 MCP(Model Context Protocol)通过标准化的接口,为 AI 模型提供了强大的实时数据接入能力,使其能够快速获取和处理来自不同数据源的实时信息。…

剑指Offer(数据结构与算法面试题精讲)C++版——day8

剑指Offer&#xff08;数据结构与算法面试题精讲&#xff09;C版——day8 题目一&#xff1a;链表中环的入口节点题目二&#xff1a;两个链表的第1个重合节点题目三&#xff1a;反转链表附录&#xff1a;源码gitee仓库 题目一&#xff1a;链表中环的入口节点 这道题的有如下三个…

【Qt】QxOrm:下载、安装、使用

1、下载源码 github地址:https://github.com/QxOrm/QxOrm 稳定版本下载:https://github.com/QxOrm/QxOrm/releases/tag/1.5.0 2、编译源码 QxOrm支持cmake编译(CMakeLists.txt)、Qt pro工程编译(QxOrm.pro) 以 QxOrm.pro 为例,编译生成的库,没有在 build-QxOrm-1.5…