一、some例题分析
1.
3176 -- Cow Bowlinghttp://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)https://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的寄存器中,以加快其存储速度