nyoj 括号匹配(二) 【DP】

括号匹配(二)

时间限制:1000 ms  |  内存限制:65535 KB

难度:6

描述
给你一个字符串,里面只包含"(",")","[","]"四种符号,请问你需要至少添加多少个括号才能使这些括号匹配起来。

如:

[]是匹配的

([])[]是匹配的

((]是不匹配的

([)]是不匹配的

输入
第一行输入一个正整数N,表示测试数据组数(N<=10)

每组测试数据都只有一行,是一个字符串S,S中只包含以上所说的四种字符,S的长度不超过100

输出
对于每组测试数据都输出一个正整数,表示最少需要添加的括号的数量。每组测试输出占一行
样例输入
4
[]
([])[]
((]
([)]
样例输出
0
0
3
2

分析:我们用d[i][j]来表示从i到j之间要插入的个数。

则有以下情况:当s[i]与s[j]配对的时候此时有dp[i][j] = min(dp[i][j], dp[i+1][j-1]),之后呢就对i和j进行区间分割比较。

代码1(普通):

#include <stdio.h>
#include <string.h>
#include <algorithm>
#define INF 0x3f3f3f3f
#define M 105
using namespace std;

char s[M];
int dp[M][M];
int main(){
    int t;
    scanf("%d", &t);
    while(t --){
        scanf("%s", s);
        memset(dp, 0, sizeof(dp));
        int i, len = strlen(s), j;
        for(i = 0; i < len; i ++)
            dp[i][i] = 1;
        for(int l = 1; l < len; l ++){
            for(i= 0;i < len-l; i ++){
                j = l+i;
                dp[i][j] = INF;
                if((s[i] == '['&&s[j] == ']')||(s[i] == '('&&s[j] == ')'))
                    dp[i][j] = min(dp[i][j], dp[i+1][j-1]);
                for(int m = i; m < j; m ++)
                    dp[i][j] = min(dp[i][j], dp[i][m]+dp[m+1][j]);
            }
        }
        printf("%d\n", dp[0][len-1]);
    }
    return 0;
}
        

代码2(记忆化dfs(跟skiing差不多的方式)):

#include <stdio.h>
#include <string.h>
#include <algorithm>
#define INF 0x3f3f3f3f
#define M 105
using namespace std;

char s[M];
int d[M][M];

int f(int x, int y){
    if(x > y) return 0;
    if(d[x][y] != -1) return d[x][y];
    if(x == y){
        d[x][x] = 1; return d[x][x];
    }
    int i, va = INF;
    if((s[x] == '('&&s[y] == ')')||(s[x] == '['&&s[y] == ']'))
    va = min(va, f(x+1, y-1));
    for(i = x; i < y; i ++){
        va = min(va, f(x, i)+f(i+1, y));
    }
    return d[x][y] = va;
}
int main(){
    int t;
    scanf("%d", &t);
    while(t --){
        scanf("%s", s);
        memset(d, -1, sizeof(d));
        printf("%d\n", f(0, strlen(s)-1));
    }
    return 0;
}
时间: 2024-10-03 11:36:24

nyoj 括号匹配(二) 【DP】的相关文章

NYOJ 括号匹配系列2,5

本文出自:http://blog.csdn.net/svitter 括号匹配一:http://acm.nyist.net/JudgeOnline/problem.php?pid=2 括号匹配二:http://acm.nyist.net/JudgeOnline/problem.php?pid=15 之前被这个题目难住,现在看动态规划就顺便过来AC了它.结果发现当年被难住一点也不丢人.. 括号匹配一很简单,就是栈的应用,AC代码: //================================

nyoj 括号匹配

这个方程有两种形式,本文采用 if(s[i]=s[j]) dp[i][j]=d[i-1][j-1] dp[i][j]=min(dp[i][k]+dp[k+1][j],dp[i][j]) (i=<k<j) 其实与另一种方法比较:根据j的所有匹配情况取最小值 1.i到j无匹配,取为dp[i][j-1]+1 2.列举所有匹配情况 dp[i][k-1]+dp[k+1][j] 取上述所有情况最小值 两者都能获得正确的结果. 同时两者的初始化为 dp[i][j]==1 if(i==j) 规划方向为:  

NYOJ - 括号匹配(二)(经典dp)

括号匹配(二) 时间限制:1000 ms | 内存限制:65535 KB 难度:6 描写叙述 给你一个字符串,里面仅仅包括"(",")","[","]"四种符号,请问你须要至少加入多少个括号才干使这些括号匹配起来. 如: []是匹配的 ([])[]是匹配的 ((]是不匹配的 ([)]是不匹配的 输入 第一行输入一个正整数N,表示測试数据组数(N<=10) 每组測试数据都仅仅有一行,是一个字符串S,S中仅仅包括以上所说的四

hihoCoder 1300 展胜地的鲤鱼旗 (括号匹配问题 dp)

#1300 : 展胜地的鲤鱼旗 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 岩手县北上市的「北上市立公园展胜地」,是陆奥国三大樱花名所之一.每年的四月中旬到五月初,这里都会举办盛大的祭奠.除了可以在盛开的樱花步道上乘坐观光马车徐行.还有横跨北上川上的鲤鱼旗,河畔还有当地特有的为祭奠祖先而编创的北上鬼剑舞. 假设,我们用一个包含 '(', ')'的括号字符串来区别每面鲤鱼旗的方向.一段括号序列被称为合法的,当且仅当满足两个条件:一.对于整个序列,左括号数量等于右括

poj 2955 Brackets 括号匹配 区间dp

题意:最多有多少括号匹配 思路:区间dp,模板dp,区间合并. 对于a[j]来说: 刚開始的时候,转移方程为dp[i][j]=max(dp[i][j-1],dp[i][k-1]+dp[k][j-1]+2), a[k]与a[j] 匹配,结果一组数据出错 ([]]) 检查的时候发现dp[2][3]==2,对,dp[2][4]=4,错了,简单模拟了一下发现,dp[2][4]=dp[2][1]+dp[2][3]+2==4,错了 此时2与4已经匹配,2与3已经无法再匹配. 故转移方程改为dp[i][j]=

imooc数据结构探险-栈篇 栈应用括号匹配二 由群友启发改良james_yuan老师算法

如图所示 引用群友内容 //老师代码有点麻烦了,不用声明两个mystack的,直接判断是否是左括号, //是的话就在mystack中push另一半括号:如果是右括号且又不是需要的括号, //就直接打印不匹配,如果是需要的右括号,就pop掉左括号.最后看mystack中栈顶是否为0,为0则打印括号匹配 /* 老师最后一点其实错了,default 其实没有必要写,只要把pNeedStack->pop(currentNeed)改为 if(!pNeedStack->pop(currentNeed))

[poj2955]括号匹配(区间dp)

解题关键:了解转移方程即可. 转移方程:$dp[l][r] = dp[l + 1][r - 1] + 2$ 若该区间左右端点成功匹配.然后对区间内的子区间取max即可. 1 #include<cstdio> 2 #include<cstring> 3 #include<algorithm> 4 #include<cstdlib> 5 #include<cmath> 6 #include<iostream> 7 using namesp

poj2955括号匹配 区间DP

类似于上一篇博文. #include<stdio.h> #include<string.h> const int maxn = 120; char s[maxn]; int dp[maxn][maxn]; int max(int x,int y) { return x>y?x:y; } int main() { int i,j,k; while(scanf("%s",s)) { if(strcmp(s,"end")==0) break;

NYOJ 15 括号匹配(二) dp

题目连接:check here~ 题意是说给一个字符串,包含'(',')','[',']'四种字符,判断至少需要添加几个字符使所给字符串括号匹配. 区间型动态规划,设dp[i][j]表示在字符串s中i位置到j位置所需要添加的最少的字符(i <= j) 有两种情况: 1.dp[i][j] = dp[i+1][j] + 1; 表示:在i到j之间没有与s[i]相匹配的括号,则必须添加一个字符来与之匹配,问题就转化为:从i+1位置到j位置所需要添加的最少的字符+1. 2.dp[i][j] = min{