当前位置: 首页 > news >正文

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


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

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

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

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

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

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

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

目录

粉刷房子

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


粉刷房子

粉刷房子

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

分析:

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]);
    }
};

顺利通过~


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

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

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


相关文章:

  • 【辰辉创聚生物】提供上万种单抗/多抗及其偶联物
  • 程序加壳脱壳原理和实现
  • P3367 【模板】并查集
  • 【局域网】
  • 记 etcd 无法在docker-compose.yml启动后无法映射数据库目录的问题
  • 视觉目标检测大模型GAIA
  • HTTP:三.HTTP报文
  • Win11系统 VMware虚拟机 安装教程
  • mac|使用scrcpy实现无线Android投屏
  • android TabLayout中tabBackground和background的区别
  • SSRF打靶总结
  • JS 构造函数实现封装性
  • 从keys到SCAN:Redis批量删除的进化之路
  • BT面板docker搭建excalidraw遇到的问题
  • AI大模型从0到1记录学习 day16
  • 水利水电安全员C证怎么考,报考有什么流程
  • Franka 机器人x Dexterity Gen引领遥操作精细任务新时代
  • 从三围学校项目看:中建海龙智能建造的崛起与突破
  • 安装python -m weditor报错解决方法
  • 7. 解立方根
  • 合肥集团网站建设/黄冈网站建设收费
  • 房屋租赁系统网站开发/seo优化推广教程
  • 电商网站的支付模块怎么做/湖北seo服务
  • 视频网站模板下载/成都企业seo
  • 天天联盟没网站怎么做/百度竞价排名展示方式
  • 网站地区词优化/百度灰色关键词排名