括号匹配问题(区间dp)
简单的检查括号是否配对正确使用的是栈模拟,这个不必再说,现在将这个问题改变一下:如果给出一个括号序列,问需要把他补全成合法最少需要多少步?
这是一个区间dp问题,我们可以利用区间dp来解决,直接看代码吧!
/* ***********************************************
Author :xiaowuga
Created Time :2017年10月03日 星期二 14时20分04秒
File Name :Desktop/text.cpp
************************************************ */
#include <bits/stdc++.h>
#define mem(s,ch) memset(s,ch,sizeof(s))
typedef long long LL;
#define inf 0x3f3f3f3f
const long long N=;
const long long mod=1e9+;
#define endl '\n'
using namespace std;
bool check(char a,char b){
if((a=='('&&b==')')||(a=='['&&b==']')) return true;
else return false;
}
int dp[][];
string q;
int main(){
ios::sync_with_stdio(false);cin.tie();
int n;
cin>>n;
while(n--){
cin>>q;
int len=q.size();
memset(dp,,sizeof(dp));
for(int i=;i<len;i++) dp[i][i]=;
for(int i=len-;i>=;i--){
for(int j=i+;j<len;j++){
dp[i][j]=inf;
if(check(q[i],q[j])) dp[i][j]=dp[i+][j-];
for(int k=i;k<j;k++){
dp[i][j]=min(dp[i][j],dp[i][k]+dp[k+][j]);
}
}
}
cout<<dp[][len-]<<endl;
}
return ;
}
虽然也有记忆化搜索的做法,但是刷表的方法似乎代码量更短,区间dp的刷表有一个特点i和j总是反过来,这是为什么呢?由于区间dp中大区间的答案依赖于小区间,所以我们在更新大区间答案的时候一定要保证向他发生状态转移的小区间都已经得出了答案。所以:
for(int i=len-2;i>=0;i--){
for(int j=i;j<len;j++){
}
}
自己枚举一下更新区间的顺序就会发现,当我们需要更新区间dp[i][j]的答案的时候dp[i][k-1]和dp[k+1][j]都已经更新了。
现在来解释一下:两种转移方式
1.如果q[i]和q[j]是可以匹配的,dp[i][j]=dp[i-1][j-1](显然不解释了)
2.对于一个区间,我们考虑把他分解成两个子区间,那么如果得到最优的子区间分配呢?答案是枚举,枚举所有的分配方式即
<[i,i],[i+1][j]>,<[i,i+1],[i+2,j]>,<[i,i+2],[i+3,j]>,<[i,i+3],[i+4,j]>,…………<[i,j-1],[j,j]>,找到所有分配方式最小值。
对于每一个区间我们找以上两种转移方式的最小值。
如果要打印打印方案(代码):
void print(int l,int r){
if(l>r) return;
if(l==r){
if(q[l]=='('||q[l]==')') cout<<"()";
else if(q[l]=='['||q[l]==']') cout<<"[]";
return ;
}
if(check(q[l],q[r])&&dp[l][r]==dp[l+][r-]){
cout<<q[l];
print(l+,r-);
cout<<q[r];
return ;
}
for(int k=l;k<r;k++){
if(dp[l][r]==dp[l][k]+dp[k+][r]){
print(l,k);
print(k+,r);
return ;
}
}
}
这个这里相当于一个逆过程,如果你看了记忆化搜索代码应该就会很好理解这个代码了
记忆化搜索:
int dp(int x,int y){
int &ans=d[x][y];
if(x>y) return ;
if(ans>||x==y) return ans;
ans=n;
if(match(str[x],str[y])) ans=min(ans,dp(x+,y-));
for(int i=x;i<y;i++){
ans=min(ans,dp(x,i)+dp(i+,y));
}
return ans;
}
练习题:
http://acm.nyist.net/JudgeOnline/problem.php?pid=15
https://vjudge.net/problem/UVA-1626
括号匹配问题(区间dp)的更多相关文章
- POJ 2955 括号匹配,区间DP
题意:给你一些括号,问匹配规则成立的括号的个数. 思路:这题lrj的黑书上有,不过他求的是添加最少的括号数,是的这些括号的匹配全部成立. 我想了下,其实这两个问题是一样的,我们可以先求出括号要匹配的最 ...
- poj2955:括号匹配,区间dp
题目大意: 给一个由,(,),[,]组成的字符串,其中(),[]可以匹配,求最大匹配数 题解:区间dp: dp[i][j]表示区间 [i,j]中的最大匹配数 初始状态 dp[i][i+1]=(i,i+ ...
- POJ 2955 Brackets --最大括号匹配,区间DP经典题
题意:给一段左右小.中括号串,求出这一串中最多有多少匹配的括号. 解法:此问题具有最优子结构,dp[i][j]表示i~j中最多匹配的括号,显然如果i,j是匹配的,那么dp[i][j] = dp[i+1 ...
- POJ - 2955 Brackets括号匹配(区间dp)
Brackets We give the following inductive definition of a “regular brackets” sequence: the empty sequ ...
- 括号序列(区间dp)
括号序列(区间dp) 输入一个长度不超过100的,由"(",")","[",")"组成的序列,请添加尽量少的括号,得到一 ...
- 九度OJ 1153:括号匹配问题 (DP)
时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:5193 解决:2248 题目描述: 在某个字符串(长度不超过100)中有左括号.右括号和大小写字母:规定(与常见的算数式子一样)任何一个左括 ...
- NYOJ15-括号匹配(二)-区间DP
pid=15">http://acm.nyist.net/JudgeOnline/problem.php? pid=15 dp[i][j]表示从i到j至少须要加入多少个括号才干满足匹配 ...
- 区间DP 基本题集
51 Nod 1021 石子归并 模板题,敲就完事了,注意一下这种状态转移方程有个四边形的优化(时间) #include <cstdio> #include <iostream> ...
- 集训第五周动态规划 J题 括号匹配
Description We give the following inductive definition of a “regular brackets” sequence: the empty s ...
- poj2955括号匹配 区间DP
Brackets Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 5424 Accepted: 2909 Descript ...
随机推荐
- RelativeLayout用代码兑现布局
RelativeLayout用代码实现布局TextView txt1 = new TextView(this); RelativeLayout.LayoutParams params = n ...
- delphi中设置listview行高的方法
第一步.在form中放置一个ImageList: 第二步.将ListView的SmallImages设置为第一步中放置的ImageList: 第三部.将imageList的height设置成自己需要的 ...
- Linux select 机制深入分析
Linux select 机制深入分析 作为IO复用的实现方式.select是提高了抽象和batch处理的级别,不是传统方式那样堵塞在真正IO读写的系统调用上.而是堵塞在sele ...
- laravel 社会化(联合)登录扩展包(QQ、微信、微博等)
laravel的官方包只支付国外网站的联合登录:http://laravelacademy.org/post/6288.html 国内用户的话,可以用这个:https://github.com/ove ...
- 盒子模型 W3C中和IE中盒子的总宽度分别是什么
W3C盒模型 总宽度 = margin-left + border-left + padding-left + width + padding-right + border-right + margi ...
- php学习四:数组(一)
1. 直接赋值方式: ① 索引数组:以索引来存储数据,内存不是连续的,但是js中的内存是连续的 代码如下: $array[0] = "11"; $array[1] = &qu ...
- 数据提交方式:post和get
众所周知,在B/S应用程序中,前台与后台的数据交互,都是通过HTML中Form表单完成的.而Form提供了两种数据传输的方式——get和post. Get请求表示客户端请求一个ur ...
- poj_2286 IDA*
题目大意 给定一个由数字组成的#字型网格,和一定的移动规则,问最少需要多少次移动才能达到要求的结果. 题目分析 要求最少需要几步到达结果,可以考虑广度优先搜索算法,或者迭代加深深度优先搜索(IDA*) ...
- Go基础---->go的基础学习(五)
这里是go中关于io的一些知识.有时不是你装得天衣无缝,而是我愿意陪你演得完美无缺. go中关于io的使用 一.Reader中的Read方法 Read 用数据填充指定的字节 slice,并且返回填充的 ...
- Express 框架的安装
从零开始用 Node.js 实现一个微博系统,功能包括路由控制.页面模板.数据库访问.用户注册.登录.用户会话等内容. Express 框架. MVC 设计模式. ejs 模板引擎 MongoDB 数 ...