给你一些左右括号,每次你可以选择一段区间将这段区间括号左变右,右变左,问最少几次可以使得整个括号序列合法。
题解:
要使得括号序列合法,则充要条件为每个位置,前面的净左括号(左括号个数-右括号个数)都要>=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;
}