湖南师范大学2018年大学生程序设计竞赛新生赛—I.巨巨的提问 括号匹配DP

给你一些左右括号,每次你可以选择一段区间将这段区间括号左变右,右变左,问最少几次可以使得整个括号序列合法。

 

题解:

要使得括号序列合法,则充要条件为每个位置,前面的净左括号(左括号个数-右括号个数)都要>=0,且最后的位置时,净左括号个数为0。

所以我们先定义d[i][j]表示当前下标为i,j为当前净左括号个数。然而要表示翻转,我们加一维表示当前这位有没有翻转即可。

所以最终状态为d[i][j][2],转移比较容易想到。

#include <iostream>
#include <cstdio>
#include <map>
#include <queue>
#include <algorithm>
#include <cstring>
#include <string>
#include <cmath>
#include <vector>
#include <assert.h>
#define IO ios::sync_with_stdio(false),cin.tie(0);
#define pb push_back
#define pii pair<int,int>
#define mp make_pair
#define rep(i,a,b) for(int i=a;i<=b;i++)
#define dep(i,a,b) for(int i=a;i>=b;i--)
#define mem(a,b) memset(a,b,sizeof(a))
#define debug(x) cout<<#x<<" = "<<x<<endl;
#define lson id<<1,l,mid
#define rson id<<1|1,mid+1,r
#define all(x) x.begin(),x.end()
typedef long long ll;
using namespace std;
const double eps = 1e-7;
const int MOD = 1e9+7;
const ll INFLL = 0x3f3f3f3f3f3f3f3f;
const int INF = 0x3f3f3f3f;
const int MAXN = 2e3 + 5;

char s[MAXN];
int d[MAXN][MAXN][2];


int main() {
#ifdef LOCAL
    freopen("input.txt", "r", stdin);
#endif
    int n;
    mem(d, 0x3f);
    scanf("%d %s", &n, s + 1);
    d[0][0][0] = 0;
    for(int i = 1; i <= n; i++) {
        for(int j = i; j >= 0; j--) {
            if(s[i] == '(') {
                d[i][j][0] = min(d[i - 1][j - 1][0], d[i - 1][j - 1][1]);
                d[i][j][1] = min(d[i - 1][j + 1][0] + 1, d[i - 1][j + 1][1]);
            } else {
                d[i][j][0] = min(d[i - 1][j + 1][0], d[i - 1][j + 1][1]);
                d[i][j][1] = min(d[i - 1][j - 1][0] + 1, d[i - 1][j - 1][1]);
            }
        }
    }
    printf("%d\n", min(d[n][0][0], d[n][0][1]));
    return 0;
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值