poj 1141(区间dp+打印路径)
2016-04-14 10:59
411 查看
题意:
定义合法的括号序列如下:
1 空序列是一个合法的序列
2 如果S是合法的序列,则(S)和[S]也是合法的序列
3 如果A和B是合法的序列,则AB也是合法的序列
例如:下面的都是合法的括号序列
(), [], (()), ([]), ()[], ()[()]
下面的都是非法的括号序列
(, [, ), )(, ([)], ([(]
给定一个由'(', ')', '[', 和 ']' 组成的序列,找出以该序列为子序列的最短合法序列。
解题思路:这题只是普通的区间dp多了一个路径输出的过程,dp[i][j]表示i-j要匹配需要修改的最小值。输出路径的时候只要根据状态转移方程就可以确定了。这里有一个要注意的地方,如果说i和j是(和),dp[i][j]不一定就是dp[i+1][j-1],因为i和j不一定就是最后要匹配的,i,j可能和中间的括号匹配。例如:[][]
详细的见代码:
#include<iostream>
#include<cstdio>
#include<cstring>
#define min(a,b) a < b ? a : b
using namespace std;
const int maxn = 105;
const int inf = 0x7fffff;
int n,dp[maxn][maxn],c[maxn][maxn];
char str[maxn];
void Path(int l,int r)
{
if(l > r) return;
else if(l == r)
{
if(str[l] == '(' || str[l] == ')')
printf("()");
else if(str[l] == '[' || str[l] == ']')
printf("[]");
return;
}
if((str[l] == '(' && str[r] == ')' || str[l] == '[' && str[r] == ']') && dp[l][r] == dp[l+1][r-1])
{
printf("%c",str[l]);
Path(l+1,r-1);
printf("%c",str[r]);
}
else
{
for(int k = l; k < r; k++)
if(dp[l][r] == dp[l][k] + dp[k+1][r])
{
Path(l,k);
Path(k+1,r);
return;
}
}
}
int main()
{
while(gets(str+1))
{
n = strlen(str+1);
memset(dp,0,sizeof(dp));
for(int i = 1; i <= n; i++)
dp[i][i] = 1;
for(int l = 2; l <= n; l++)
for(int i = 1; i <= n; i++)
{
int j = i + l - 1;
dp[i][j] = inf;
if(j > n) break;
if(str[i] == '(' && str[j] == ')' || str[i] == '[' && str[j] == ']')
dp[i][j] = dp[i+1][j-1];
for(int k = i; k < j; k++)
dp[i][j] = min(dp[i][j],dp[i][k]+dp[k+1][j]);
}
Path(1,n);
printf("\n");
}
return 0;
}
定义合法的括号序列如下:
1 空序列是一个合法的序列
2 如果S是合法的序列,则(S)和[S]也是合法的序列
3 如果A和B是合法的序列,则AB也是合法的序列
例如:下面的都是合法的括号序列
(), [], (()), ([]), ()[], ()[()]
下面的都是非法的括号序列
(, [, ), )(, ([)], ([(]
给定一个由'(', ')', '[', 和 ']' 组成的序列,找出以该序列为子序列的最短合法序列。
解题思路:这题只是普通的区间dp多了一个路径输出的过程,dp[i][j]表示i-j要匹配需要修改的最小值。输出路径的时候只要根据状态转移方程就可以确定了。这里有一个要注意的地方,如果说i和j是(和),dp[i][j]不一定就是dp[i+1][j-1],因为i和j不一定就是最后要匹配的,i,j可能和中间的括号匹配。例如:[][]
详细的见代码:
#include<iostream>
#include<cstdio>
#include<cstring>
#define min(a,b) a < b ? a : b
using namespace std;
const int maxn = 105;
const int inf = 0x7fffff;
int n,dp[maxn][maxn],c[maxn][maxn];
char str[maxn];
void Path(int l,int r)
{
if(l > r) return;
else if(l == r)
{
if(str[l] == '(' || str[l] == ')')
printf("()");
else if(str[l] == '[' || str[l] == ']')
printf("[]");
return;
}
if((str[l] == '(' && str[r] == ')' || str[l] == '[' && str[r] == ']') && dp[l][r] == dp[l+1][r-1])
{
printf("%c",str[l]);
Path(l+1,r-1);
printf("%c",str[r]);
}
else
{
for(int k = l; k < r; k++)
if(dp[l][r] == dp[l][k] + dp[k+1][r])
{
Path(l,k);
Path(k+1,r);
return;
}
}
}
int main()
{
while(gets(str+1))
{
n = strlen(str+1);
memset(dp,0,sizeof(dp));
for(int i = 1; i <= n; i++)
dp[i][i] = 1;
for(int l = 2; l <= n; l++)
for(int i = 1; i <= n; i++)
{
int j = i + l - 1;
dp[i][j] = inf;
if(j > n) break;
if(str[i] == '(' && str[j] == ')' || str[i] == '[' && str[j] == ']')
dp[i][j] = dp[i+1][j-1];
for(int k = i; k < j; k++)
dp[i][j] = min(dp[i][j],dp[i][k]+dp[k+1][j]);
}
Path(1,n);
printf("\n");
}
return 0;
}
相关文章推荐
- 详解Android应用中屏幕尺寸的获取及dp和px值的转换
- 基于Android中dp和px之间进行转换的实现代码
- Android中dip、dp、sp、pt和px的区别详解
- LFC1.0.0 版本发布
- Android px、dp、sp之间相互转换
- HP data protector软件学习1--基本角色与基本工作流程
- HP data protector软件学习2--软件组成与界面介绍
- android中像素单位dp、px、pt、sp的比较
- Android对px和dip进行尺寸转换的方法
- Android根据分辨率进行单位转换-(dp,sp转像素px)
- android 尺寸 dp,sp,px,dip,pt详解
- DP问题各种模型的状态转移方程
- POJ-1695-Magazine Delivery-dp
- nyoj-1216-整理图书-dp
- TYVJ1193 括号序列解题报告
- 对DP的一点感想
- TYVJ上一些DP的解题报告
- soj1005. Roll Playing Games
- 01背包问题
- LeetCode之Maximum Product Subarray