poj 2955 Brackets(区间DP求最长匹配子串)

思路:假设要求区间[i,j]的最长匹配字串,它必然可以从[i,j-1]转移而来,有可能是s[j]与s[i]发生“关系”(匹配或不匹配),一直到s[j-1],若不发生“关系”,即s[j]跟自己发生“关系”,用for循环枚举所有的可能,取最大值。

代码:

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;

char s[105];
int dp[105][105];//代表[i,j]区间的最长匹配子串

int check(int i,int j)
{
    if(s[i]=='('&&s[j]==')')
        return 1;
    if(s[i]=='['&&s[j]==']')
        return 1;
    return 0;
}

int main()
{
    while(scanf("%s",s)&&strcmp(s,"end")!=0)
    {
        memset(dp,0,sizeof(dp));
        int len=strlen(s);
        for(int i=0;i<len;i++)
        {
            dp[i][i]=0;
            if(check(i,i+1))
            {
                dp[i][i+1]=2;
            }
        }
        for(int i=3;i<=len;i++)
        {
            for(int j=0;j+i-1<len;j++)
            {
                if(check(j,j+i-1))
                    dp[j][j+i-1]=dp[j+1][j+i-2]+2;
                else
                    dp[j][j+i-1]=dp[j+1][j+i-2];
                for(int k=j+1;k<=j+i-1;k++)
                {
                    dp[j][j+i-1]=max(dp[j][j+i-1],dp[j][k-1]+dp[k][j+i-1]);
                }
            }
        }
        /*for(int i=0;i<len;i++)
        {
            for(int j=0;j<len;j++)
            {
                if(j==len-1)
                    printf("%d\n",dp[i][j]);
                else if(j<i)
                    printf("  ");
                else
                    printf("%d ",dp[i][j]);
            }
        }*/
        printf("%d\n",dp[0][len-1]);
    }
    return 0;
}
时间: 2024-10-10 04:19:29

poj 2955 Brackets(区间DP求最长匹配子串)的相关文章

POJ 2955 Brackets (区间dp 括号匹配)

Brackets Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 3951   Accepted: 2078 Description We give the following inductive definition of a "regular brackets" sequence: the empty sequence is a regular brackets sequence, if s is a reg

POJ 2955 Brackets (区间DP)

题意:给定一个序列,问你最多有多少个合法的括号. 析:区间DP,dp[i][j] 表示在 第 i 到 第 j 区间内最多有多少个合法的括号. 代码如下: #pragma comment(linker, "/STACK:1024000000,1024000000") #include <cstdio> #include <string> #include <cstdlib> #include <cmath> #include <ios

POJ 2955 Brackets 区间DP 最大括号匹配

http://blog.csdn.net/libin56842/article/details/9673239 http://www.cnblogs.com/ACMan/archive/2012/08/09/2630497.html http://blog.csdn.net/chaiyuan414/article/details/5448699 #include <iostream> #include <string> #include <cstring> #inclu

POJ 2774 后缀数组:求最长公共子串

思路:其实很简单,就是两个字符串连接起来,中间用个特殊字符隔开,然后用后缀数组求最长公共前缀,然后不同在两个串中,并且最长的就是最长公共子串了. 注意的是:用第一个字符串来判断是不是在同一个字符中,刚开始用了第二个字符的长度来判断WA了2发才发现. #include<iostream> #include<cstdio> #include<cstring> #include<algorithm> #include<map> #include<

Brackets POJ - 2955 (区间dp)

Brackets POJ - 2955 题意:给一个括号序列,问最多有多少个括号是可以配对的. 1 #include<cstdio> 2 #include<algorithm> 3 #include<string> 4 #include<iostream> 5 #include<cstring> 6 using namespace std; 7 const int maxn=110; 8 int dp[maxn][maxn]; 9 char s[

POJ 2955 Brackets 区间合并

输出一个串里面能匹配的括号数 状态转移方程: if(s[i]=='('&&s[j]==')'||s[i]=='['&&s[j]==']')             dp[i][j]=dp[i+1][j-1]+2; 然后再区间合并 1 //#pragma comment(linker, "/STACK:167772160")//手动扩栈~~~~hdu 用c++交 2 #include<cstdio> 3 #include<cstring&

poj 2774 字符串哈希求最长公共子串

Long Long Message #include <iostream> #include <algorithm> #include <cstdio> #include <cstdlib> #include <cstring> #include <queue> #include <stack> #include <cmath> #include <map> #include <string&

POJ 2955 Brackets (动规)

Brackets Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 2999   Accepted: 1536 Description We give the following inductive definition of a "regular brackets" sequence: the empty sequence is a regular brackets sequence, if s is a reg

POJ 2955 Brackets(计数问题吗呵呵)

我只能说这道题和上一道动态规划的问题真的是太像了,连方法也一模一样 确实,计数也需要存状态,计数也是需要动规的. 此时d[i][j]表示的状态是s[i~j]的序列中有多少 不规则 的括号. #include<cstdio> #include<cstring> #include<algorithm> using namespace std; int n; char s[105]; int d[105][105]; bool match(char ch1,char ch2)