区间dp入门+例题
区间dp作为线性dp的一种,顾名思义是以区间作为阶段进行dp的,使用它的左右端点描述每个维度,决策往往是从小状态向大状态转移中推得的。它跟st表等树状结构有着相似的原理---向下划分,向上递推。
dp最终要求的就是推出状态转移方程,从板子中我们可以感受出来区间dp的关键在于如何找到小状态与大状态的关系。
for(int i=;i<n;i++){//区间长度
for(int l=;l+i<=n;l++){//左端点
for(int h=l;h<l+i;h++)//枚举区间合并的分割点找到最优解
//转移方程
}
}
这样基本的板子时间复杂度会达到o(n^3),如果被卡的话通常就要从四边形不等式等状态转移的性质出发能不能找到更好的转移,优化掉冗余。dp是拿空间换取时间的搜索,只要优化掉冗余推平不是梦。(蒟蒻口胡orz~)
刷了一点蓝书上的例题,分享一下存个档
石子合并
非常经典的区间dp题,肉眼可见dp[l][r]=minrk=l(dp[l][r],dp[l][k]+dp[k+1][r])
#include<bits/stdc++.h>
using namespace std;
int dp[][];
int sum[];
int main()
{
int n;scanf("%d",&n);
memset(dp,0x3f,sizeof dp);
for(int i=;i<=n;i++)
scanf("%d",&dp[i][]),sum[i]=sum[i-]+dp[i][],dp[i][i]=;
for(int i=;i<n;i++){//区间长度
for(int l=;l+i<=n;l++){//左端点
for(int h=l;h<l+i;h++)//枚举区间合并的分割点找到最优解
dp[l][l+i]=min(dp[l][h]+dp[h+][l+i]+sum[l+i]-sum[l-],dp[l][l+i]);//转移方程
}
}
cout<<dp[][n]<<endl;
}
Polygon
枚举删掉的第一条边,就跟上一题类似了,由于存在乘,同时维护最大和最小的状态转移一下就OK啦
#include<bits/stdc++.h>
using namespace std;
char c[][];
long long dp[][][],num[];
vector <int> v,ans;
int main()
{
int n;scanf("%d",&n);getchar();
for(int i=;i<=*n;i++){
if(i&){
int t1=i/,t2=i/+;
if(t1==) t1=n;
scanf("%c",&c[t1][t2]);
c[t2][t1]=c[t1][t2];
}
else scanf("%lld",&num[i/]),getchar();
}
int tot=,mi=-1e9;
for(int i=;i<=n;i++){
v.clear();v.push_back();
for(int j=i;j<=n;j++)
v.push_back(j);
for(int j=;j<i;j++)
v.push_back(j);
for(int j=;j<=n;j++)
for(int h=;h<=n;h++)
dp[j][h][]=-1e18,dp[j][h][]=1e18;
for(int j=;j<=n;j++) dp[j][j][]=dp[j][j][]=num[v[j]];
for(int j=;j<n;j++){
for(int l=;l+j<=n;l++){
for(int k=l;k<l+j;k++){
long long t1=1e18,t2=-1e18;
if(c[v[k]][v[k+]]=='t')
t1=min(t1,dp[l][k][]+dp[k+][l+j][]),
t2=max(t2,dp[l][k][]+dp[k+][l+j][]);
else{
for(int p=;p<;p++)
for(int q=;q<;q++)
t1=min(t1,dp[l][k][p]*dp[k+][l+j][q]),
t2=max(t2,dp[l][k][p]*dp[k+][l+j][q]);
}
dp[l][l+j][]=max(dp[l][l+j][],t2);
dp[l][l+j][]=min(dp[l][l+j][],t1);
}
}
}
if(dp[][n][]>mi){
mi=dp[][n][];ans.clear();ans.push_back(tot);
}
else if(dp[][n][]==mi) ans.push_back(tot);
tot++;
}
cout<<mi<<endl;
int l=ans.size();
for(int i=;i<l;i++)
printf("%d%c",ans[i],i==l-?'\n':' ');
}
当然这个还有优化成o(n^3)的写法,对枚举删第一条进行优化。
金字塔
阅读题。。。dp[l][r]表示s[l]-s[r]的可以构成的个数,那么枚举k为第一个子树的分割点就可以得出
dp[l][r]=Σdp[l][k-1]+dp[k][r-1]
#include<bits/stdc++.h>
using namespace std;
const int maxn=3e2+;
const int mod=1e9;
char s[maxn];
int dp[maxn][maxn];
int main()
{
scanf("%s",s+);int n=strlen(s+);
for(int i=;i<=n;i++) dp[i][i]=;
for(int l=n-;l>=;l--)
for(int r=l+;r<=n;r++){
if(s[l]==s[r]){
for(int k=l+;k<r;k++)
dp[l][r]=(dp[l][r]+1LL*dp[l][k-]*dp[k][r-]%mod)%mod;
}
}
cout<<dp[][n]<<endl;
}
区间dp入门+例题的更多相关文章
- POJ 2955 Brackets (区间dp入门)
Description We give the following inductive definition of a “regular brackets” sequence: the empty s ...
- 【DP】区间DP入门
在开始之前我要感谢y总,是他精彩的讲解才让我对区间DP有较深的认识. 简介 一般是线性结构上的对区间进行求解最值,计数的动态规划.大致思路是枚举断点,然后对断点两边求取最优解,然后进行合并从而得解. ...
- hdu 4570 Multi-bit Trie 区间DP入门
Multi-bit Trie 题意:将长度为n(n <= 64)的序列分成若干段,每段的数字个数不超过20,且每段的内存定义为段首的值乘以2^(段的长度):问这段序列总的内存最小为多少? 思路: ...
- POJ2955--Brackets 区间DP入门 括号匹配
题意很简单,就是求给出串中最大的括号匹配数目.基础题,格式基本为简单区间dp模板. #include<iostream> #include<string.h> using na ...
- HRBUST - 1818 石子合并 区间dp入门
有点理解了进阶指南上说的”阶段,状态和决策“ /* 区间dp的基础题: 以区间长度[2,n]为阶段,枚举该长度的区间,状态dp[l][r]表示合并区间[l,r]的最小费用 状态转移方程dp[l][r] ...
- 区间DP入门题目合集
区间DP主要思想是先在小区间取得最优解,然后小区间合并时更新大区间的最优解. 基本代码: //mst(dp,0) 初始化DP数组 ;i<=n;i++) { dp[i][i]=初始 ...
- [nyoj737]石子归并(区间dp入门题)
题意:有N堆石子排成一排,每堆石子有一定的数量.现要将N堆石子并成为一堆.合并的过程只能每次将相邻的两堆石子堆成一堆,每次合并花费的代价为这两堆石子的和,经过N-1次合并后成为一堆.求出总的代价最小值 ...
- 区间DP入门
所为区间DP,主要是把一个大区间拆分成几个小区间,先求小区间的最优值,然后合并起来求大区间的最优值. 区间DP最关键的就是满足最优子结构以及无后效性!! 例如像是石子合并和括号匹配这两类比较经典的模型 ...
- poj 2955 区间dp入门题
第一道自己做出来的区间dp题,兴奋ing,虽然说这题并不难. 从后向前考虑: 状态转移方程:dp[i][j]=dp[i+1][j](i<=j<len); dp[i][j]=Max(dp[i ...
随机推荐
- 【Weiss】【第03章】练习3.3:通过交换指针交换单/双链表元素
[练习3.3] 通过之调整指针(而不是数据)来交换两个相邻的元素,使用 a.单链表 b.双链表 Answer: 先放测试代码,折叠标题可以看到分别是哪种链表的测试. 实测可满足题意,但单链表和双链表的 ...
- Java多线程并发06——CAS与AQS
在进行更近一步的了解Java锁的知识之前,我们需要先了解与锁有关的两个概念 CAS 与 AQS.关注我的公众号「Java面典」了解更多 Java 相关知识点. CAS(Compare And Swap ...
- 数据库表结构查询SQL
今天给大家送上两个SQL查询的方法,也许在项目中你都用过,但是,没关系,仅仅记录下,以便后面不用去查找.针对与经常写SQL的同行来说,应该是非常简单的. 查询表结构数据 SELECT t.COLUMN ...
- css清除浮动影响
将清除浮动代码添加到重置样式表中,随时可以调用 }}.clearfix:after{clear:both} 给需要清除浮动影响的元素添加class名 --- clearfix 例: <!-- c ...
- TortoiseGit 与 Putty 配置冲突导致 Server refuse our key
tortoisegit是TortoiseSVN的Git版本,tortoisegit用于迁移TortoiseSVN到TortoiseGit,一直以来Git在Windows平台没有好用GUI客户端,现在t ...
- 一个完整的机器学习项目在Python中演练(三)
大家往往会选择一本数据科学相关书籍或者完成一门在线课程来学习和掌握机器学习.但是,实际情况往往是,学完之后反而并不清楚这些技术怎样才能被用在实际的项目流程中.就像你的脑海中已经有了一块块"拼 ...
- 七大Github机器学习热门项目
译者 | 小韩 来源 | analyticsvidhya.com[磐创AI导读]:让我们一起来看下近期热门的机器学习Github仓库,包括了自然语言处理(NLP).计算机视觉(CV)与大数据等多个领域 ...
- List<Object>转List<T>
今天遇到一个麻烦,公司有个项目用了一个封装dao的模板,他妈的不管是查一条数据的方法,还是查一个集合数据的方法,全都返回Object或List<Object> 由于对象是Object根本不 ...
- Java实现自定义数组及其方法
自定义数组 主要功能有增.删(根据索引,根据值).改.查扩容等功能 package array; public class CustomArray { private int[] array = nu ...
- Python语法元素分析
缩进 1个缩进 = 4个空格 用以在Python中标明代码的层次关系 缩进是Python语言中表明程序框架的唯一手段 注释 注释:程序员在代码中加入的说明信息,不被计算机执行 注释的两种方法: 单行注 ...