区间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入门+例题的更多相关文章

  1. POJ 2955 Brackets (区间dp入门)

    Description We give the following inductive definition of a “regular brackets” sequence: the empty s ...

  2. 【DP】区间DP入门

    在开始之前我要感谢y总,是他精彩的讲解才让我对区间DP有较深的认识. 简介 一般是线性结构上的对区间进行求解最值,计数的动态规划.大致思路是枚举断点,然后对断点两边求取最优解,然后进行合并从而得解. ...

  3. hdu 4570 Multi-bit Trie 区间DP入门

    Multi-bit Trie 题意:将长度为n(n <= 64)的序列分成若干段,每段的数字个数不超过20,且每段的内存定义为段首的值乘以2^(段的长度):问这段序列总的内存最小为多少? 思路: ...

  4. POJ2955--Brackets 区间DP入门 括号匹配

    题意很简单,就是求给出串中最大的括号匹配数目.基础题,格式基本为简单区间dp模板. #include<iostream> #include<string.h> using na ...

  5. HRBUST - 1818 石子合并 区间dp入门

    有点理解了进阶指南上说的”阶段,状态和决策“ /* 区间dp的基础题: 以区间长度[2,n]为阶段,枚举该长度的区间,状态dp[l][r]表示合并区间[l,r]的最小费用 状态转移方程dp[l][r] ...

  6. 区间DP入门题目合集

      区间DP主要思想是先在小区间取得最优解,然后小区间合并时更新大区间的最优解.       基本代码: //mst(dp,0) 初始化DP数组 ;i<=n;i++) { dp[i][i]=初始 ...

  7. [nyoj737]石子归并(区间dp入门题)

    题意:有N堆石子排成一排,每堆石子有一定的数量.现要将N堆石子并成为一堆.合并的过程只能每次将相邻的两堆石子堆成一堆,每次合并花费的代价为这两堆石子的和,经过N-1次合并后成为一堆.求出总的代价最小值 ...

  8. 区间DP入门

    所为区间DP,主要是把一个大区间拆分成几个小区间,先求小区间的最优值,然后合并起来求大区间的最优值. 区间DP最关键的就是满足最优子结构以及无后效性!! 例如像是石子合并和括号匹配这两类比较经典的模型 ...

  9. poj 2955 区间dp入门题

    第一道自己做出来的区间dp题,兴奋ing,虽然说这题并不难. 从后向前考虑: 状态转移方程:dp[i][j]=dp[i+1][j](i<=j<len); dp[i][j]=Max(dp[i ...

随机推荐

  1. 强连通分量SCC 2-SAT

    强连通分量SCC 2-SAT 部分资料来自: 1.https://blog.csdn.net/whereisherofrom/article/details/79417926 2.https://ba ...

  2. 开源项目OEIP 游戏引擎与音视频多媒体(UE4/Unity3D)

    现开源一个项目 OEIP 项目实现的功能Demo展示 这个项目演示了在UE4中,接入摄像机通过OEIP直接输出到UE4纹理上,并直接把UE4里的RenderTarget当做输入源通过OEIP里GPU管 ...

  3. Redis 【常识与进阶】

    Redis 简介 Redis 是完全开源免费的,遵守BSD协议,是一个高性能的key-value数据库. Redis 与其他 key - value 缓存产品有以下三个特点: Redis支持数据的持久 ...

  4. identityserver4源码解析_3_认证接口

    目录 identityserver4源码解析_1_项目结构 identityserver4源码解析_2_元数据接口 identityserver4源码解析_3_认证接口 identityserver4 ...

  5. 使用SparkSQL编写wordCount的词频统计

    # 使用SparkSQL编写wordCount的词频统计 ## word.txt```hello hello scala sparkjava sql html java hellojack jack ...

  6. 彻底明白equals和hashCode

    equals和hashCode方法 equals 我们知道equals是用来比较两个对象是否相等的,比如我们常用的String.equals方法 @Test public void test() { ...

  7. Alpine Linux 常用命令

    一:Alpine Linux开启SSH远程登陆 1.简介: 最重要的一个服务了,远程登陆需要用它,文件传输需要用它,必备功能.不管你是在实体机上跑,虚拟机上跑,docker里面跑,这个都是必须的. 2 ...

  8. MFC之创建多级动态菜单

    一开始以我是这样做的,结果是错误的: 这段代码第一次点击时,会在第6个位置创建MFC菜单,我本以为再次点击,menu->GetSubMenu(5)返回的值就不会为空了,但事实是它返回了NULL, ...

  9. 给rm命令加保险

    众所周知,脑残可以学习,但是手残没法治.相信每一位喜欢用终端操作电脑的同学都曾手误使用 rm 命令把不该删除的文件删了.然而,使用 rm 删除的文件是不会进去回收站的. 所以,最好的方法就是我们自定义 ...

  10. 重磅!!!一文总结Pytorch的8张思维导图!

    本文以思维导图的形式,为大家介绍了深度学习的核心内容,主要包括:深度学习与Pytorch简介.词向量.用pytorch处理常见的NLP和CV任务.图片风格迁移和GAN.Seq2Seq与Attentio ...