您的位置:首页 > 其它

区间dp Gym100712D Alternating Strings

2015-07-20 16:30 183 查看
地址:http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=195706

对这种字符串dp有恐惧症..

对于这题,N<=1000,所以可以支持O(n^2)的算法

对于这题,首先我们可以预处理出d[i][j],如果区间[i,j]为交替串,就是1,否则就是0

然后再设一个F[i]表示,最后一刀是在第i个和第i-1个字符的中间剪开的方法数

然后动态转移方程是这样的

if(i + k <= len && !d[i][i + k - 1]) {

F[i + k] = min(F[i + k], F[i] + 1);

}

只有当[i,i+k-1]不是交替串,且i+k<=len的时候,才可以在i+k的位置剪开

然后答案就是

int ans = INF;

for(int i = len - K + 1; i <= len; i++) {

if(!d[i][len]) ans = min(ans, F[i]);

}

因为F[i]的含义是 最后一刀是在第i个和第i-1个字符的中间剪开的方法数,刚开始我也因为没有深刻意识到F[i]的含义而错了很久

所以最后一刀可以是[len-K+1,len]里面的任意一个,但是有个前提,就是从最后一刀到最末尾的这一段不能是交替串

#include<cstdio>
#include<cmath>
#include<cstring>
#include<queue>
#include<vector>
#include<functional>
#include<algorithm>

using namespace std;
typedef long long LL;
typedef pair<int, int> PII;

const int MX = 1000 + 5;
const int INF = 0x3f3f3f3f;

char S[MX];
int F[MX];
bool d[MX][MX];

int main() {
//freopen("input.txt", "r", stdin);
int T, len, K;
scanf("%d", &T);
while(T--) {
memset(d, 0, sizeof(d));
memset(F, INF, sizeof(F));

scanf("%d%d%s", &len, &K, S + 1);

for(int L = 1; L <= len; L++) {
d[L][L] = 1;
for(int R = L + 1; R <= len; R++) {
if(d[L][R - 1] && S[R] != S[R - 1]) {
d[L][R] = 1;
} else break;
}
d[L][L] = 0;
}

F[1]=0;
for(int i = 1; i <= len; i++) {
for(int k = 1; k <= K; k++) {
if(i + k <= len && !d[i][i + k - 1]) {
F[i + k] = min(F[i + k], F[i] + 1);
}
}
}

int ans = INF;
for(int i = len - K + 1; i <= len; i++) {
if(!d[i][len]) ans = min(ans, F[i]);
}

printf("%d\n", ans);
}
return 0;
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: