括号匹配问题(区间dp)
简单的检查括号是否配对正确使用的是栈模拟,这个不必再说,现在将这个问题改变一下:如果给出一个括号序列,问需要把他补全成合法最少需要多少步?
这是一个区间dp问题,我们可以利用区间dp来解决,直接看代码吧!
1 /* *********************************************** 2 Author :xiaowuga 3 Created Time :2017年10月03日 星期二 14时20分04秒 4 File Name :Desktop/text.cpp 5 ************************************************ */ 6 #include <bits/stdc++.h> 7 #define mem(s,ch) memset(s,ch,sizeof(s)) 8 typedef long long LL; 9 #define inf 0x3f3f3f3f 10 const long long N=1000000; 11 const long long mod=1e9+7; 12 #define endl '\n' 13 using namespace std; 14 bool check(char a,char b){ 15 if((a=='('&&b==')')||(a=='['&&b==']')) return true; 16 else return false; 17 } 18 int dp[101][101]; 19 string q; 20 int main(){ 21 ios::sync_with_stdio(false);cin.tie(0); 22 int n; 23 cin>>n; 24 while(n--){ 25 cin>>q; 26 int len=q.size(); 27 memset(dp,0,sizeof(dp)); 28 for(int i=0;i<len;i++) dp[i][i]=1; 29 for(int i=len-2;i>=0;i--){ 30 for(int j=i+1;j<len;j++){ 31 dp[i][j]=inf; 32 if(check(q[i],q[j])) dp[i][j]=dp[i+1][j-1]; 33 for(int k=i;k<j;k++){ 34 dp[i][j]=min(dp[i][j],dp[i][k]+dp[k+1][j]); 35 } 36 } 37 } 38 cout<<dp[0][len-1]<<endl; 39 } 40 return 0; 41 }
虽然也有记忆化搜索的做法,但是刷表的方法似乎代码量更短,区间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]>,找到所有分配方式最小值。
对于每一个区间我们找以上两种转移方式的最小值。
如果要打印打印方案(代码):
1 void print(int l,int r){ 2 if(l>r) return; 3 if(l==r){ 4 if(q[l]=='('||q[l]==')') cout<<"()"; 5 else if(q[l]=='['||q[l]==']') cout<<"[]"; 6 return ; 7 } 8 if(check(q[l],q[r])&&dp[l][r]==dp[l+1][r-1]){ 9 cout<<q[l]; 10 print(l+1,r-1); 11 cout<<q[r]; 12 return ; 13 } 14 for(int k=l;k<r;k++){ 15 if(dp[l][r]==dp[l][k]+dp[k+1][r]){ 16 print(l,k); 17 print(k+1,r); 18 return ; 19 } 20 } 21 }
这个这里相当于一个逆过程,如果你看了记忆化搜索代码应该就会很好理解这个代码了
记忆化搜索:
1 int dp(int x,int y){ 2 int &ans=d[x][y]; 3 if(x>y) return 0; 4 if(ans>0||x==y) return ans; 5 ans=n; 6 if(match(str[x],str[y])) ans=min(ans,dp(x+1,y-1)); 7 for(int i=x;i<y;i++){ 8 ans=min(ans,dp(x,i)+dp(i+1,y)); 9 } 10 return ans; 11 }
练习题:
http://acm.nyist.net/JudgeOnline/problem.php?pid=15
https://vjudge.net/problem/UVA-1626