区间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 ...
随机推荐
- 「newbee-mall新蜂商城开源啦」GitHub 上最热门的 Spring Boot 项目,我也要做一次靓仔!
没有一个冬天不可逾越,也没有一个春天不会到来. 介绍一下新蜂商城的近况,同时,新蜂商城 Vue 版本目前也在开发中,在这篇文章里我也向大家公布一下新蜂商城 Vue 版本的开发进度,和大家同步一下,在不 ...
- Tomcat 启动过滤器异常
严重 [RMI TCP Connection(2)-127.0.0.1] org.apache.catalina.core.StandardContext.filterStart 启动过滤器异常 ja ...
- SpringMVC框架——文件的上传与下载
使用SpringMVC框架做个小练习,需求: 1.单个图片上传并显示到页面中: 2.多个图片上传并显示到页面中: 3.上传文件后下载文件: 1.pom.xml中添加依赖 <!-- 文件上传 -- ...
- SpringMVC框架——视图解析
SpringMVC视图解析,就是将业务数据绑定给JSP域对象,并在客户端进行显示. 域对象: pageContext.request.session.application 业务数据绑定是有ViewR ...
- ketika aku 病毒
#客户中了该病毒,本想找病毒样本来看看,可是没找到样本,发现中这个病毒的案例还是相对较少: #国内好像没有对于该病毒没有比较详尽的病毒信息,特此写一下方便后者: #中招表现:目前所能够发现的是能够对浏 ...
- css网页重置样式表(多版本)
Eric reset.css html, body, div, span, applet, object, iframe, h1, h2, h3, h4, h5, h6, p, blockquote, ...
- Android适配器
Android适配器 安卓的适配器在我看来是一个非常重要的知识点,面对形式相同但数据源较多的情况时,适配器是一个比较好的解决方法.数据适配器是建立了数据源与控件之间的适配关系,将数据源转换为控件能够显 ...
- Java 泛型数组问题
Java中不支持泛型数组, 以下代码会编译报错:generic array creation ArrayList<Integer>[] listArr = new ArrayList< ...
- 开发一个健壮的npm包
项目地址:loan-calculate-utils npm包的发布.更新查看上一篇文章 开发一个基础的npm包 目前我们的目录是这个样子: . ├── source 源代码目录 │ └── ind ...
- 命令行中运行Java字节码文件提示找不到或无法加载主类的问题
测试类在命令行操作,编译通过,运行时,提示 错误: 找不到或无法加载主类 java类 package com.company.schoolExercise; public class test7_3_ ...