【动态规划例题分析+总结】

一、some例题分析

1.

3176 -- Cow Bowlingicon-default.png?t=M4ADhttp://poj.org/problem?id=3176用递推代替递归,可学习的妙法:用指针取二维数组的最后一行地址,每次计算的新值覆盖原最后一行的数,实现状态的转移,每个状态中通过a[i][j]使p[j]向上一行获取新值。

2、P2001 硬币的面值

题目大概是说要组成1到m元,已给出n种类型硬币,每种硬币个数不限,求最少需要多少个硬币九能满足所有的可能价格。

思路就是贪心。一开始肯定能拼1,所以1-1是可以的,然后就找最小的a【i】>x+1,然后一直推,从而求解,注意是long long

3、P1336 最佳课题选择

题意要求,完成 nn 篇论文的最少时间是多少

f[i][j]表示前i课题中完成前j篇论文的最小花费,可得递推式f[i][j]=min{f[i-1][j-k]+a[i]*k^b[i]}

4、
URAL1513. Lemon Tale(dp)icon-default.png?t=M4ADhttps://www.cnblogs.com/shangyu/p/3543115.html n个B里不能出现超过连续k个L的情况 一维递推就可以 两种情况 1、dp[i] += dp[i-1] 在i-1的串后面直接加一个B 2、dp[i]+=dp[i-2]+dp[i-3]+...+dp[i-k-1] 这部分的意思是在串后面补连续1个L 2个L 3...K个L的情况 那补一个L的情况要取决于dp[i-1]里面最后一位为B的个数 正好为dp[i-2],依次可做。

二、区间DP

1、模板:

for(int len = 1;len<=n;len++){//枚举长度
        for(int j = 1;j+len<=n+1;j++){//枚举起点,ends<=n
            int ends = j+len - 1;
            for(int i = j;i<ends;i++){//枚举分割点,更新小区间最优解
                dp[j][ends] = min(dp[j][ends],dp[j][i]+dp[i+1][ends]+something);
            }
        }
    }

忽然就想到上次省赛的某个题,让求解区间和为固定值的区间个数,就可以用这个模板。

2、时间优化

我们可以把最优分割点保存下来,在查找的时候利用保存的最优分割点来优化查找过程。

(1)功能:用来寻找,s[i][j](i~j的最优分割点)与其他分割点的关系

(2)不等式内容:如果某东西满足a<b<=c<d且f[a][c]+f[b][d]<=f[a][d]+f[b][c],则说这个东西满足四边形不等式。简而言之:交叉小于包含!

(3)结论关系:s[i][j-1]<=s[i][j]<=s[i+1][j] 

原文链接:https://blog.csdn.net/qq_40772692/article/details/80183248

以石子归并为例,采用四边优化(不大理解)

(4)例题:

Poj2955 括号匹配(一)

(1)题意:给出一个的只有'(',')','[',']'四种括号组成的字符串,求 最多 有多少个括号满足题目里所描述的完全匹配。

(2)思路:这里的状态转移是以一个if为基础的,如果s[i]与s[j]匹配,那么明显的dp[i][j] = dp[i+1][j-1]+2;然后在这个基础上枚举分割点k.

(3)状态转移方程:dp[i][j]表示第i~j个字符间的最大匹配字符数。

if(s[i] 与 s[j]匹配) dp[i][j] = d[[i+1][j-1] +2;

dp[i][j] = max(dp[i][j],dp[i][k]+dp[k+1][j]);

Poj1651 抽卡片

(1)题意:给你n个数字,要求不能删除两端点的数字,然后删除其他数字的代价是该数字和左右相邻数字的乘积,问把数字(除端点)删完后的最小总代价。

(2)思路:因为最后都要删掉中间所有的数字,所以我们分隔一个个小区间删数字,合并区间求最小。那么我们的状态就是目前删掉的数字区间,但是我们分割的时候的意思是抽一个卡片出来,所以这个卡片不能在已经抽出的状态里面,所以dp[i][j]里面是不包含j卡片的!

(3)状态转移方程:dp[i][j]表示抽出第i~j-1张卡片时候的最小值

dp[i][j] = min(dp[i][j],dp[i][k] + dp[k+1][j] +num[i-1]*num[k]*num[j]);

整数划分
(1)题意:给出两个整数 n , m ,要求在 n 中加入m - 1 个乘号,将n分成m段,求出这m段的最大乘积

(2)思路:这里给的乘号是有限个,所以状态方程里必须包含使用乘号的个数,此外还要包含区间长度。所以怎么用二维dp实现包含m和n,我们可以用dp[i][j]表示在第1~i个字符里插入j个乘号的最大值。

(3)状态转移方程 dp[i][j]表示在第1~i个字符里插入j个乘号的最大值;用num[i][j]表示第i~j个字符表示的数字;

dp[i][j] = max(dp[i][j],dp[k][j-1]*num[k+1][i])
 

三、新发现

register修饰符暗示编译程序相应的变量将被频繁地使用,如果可能的话,应将其保存在CPU的寄存器中,以加快其存储速度

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

迎风809

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值