区间dp Gym100712D Alternating Strings

来源:互联网 发布:社交网络囤积症 编辑:程序博客网 时间:2024/05/23 13:45

地址: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;}


0 0
原创粉丝点击