码迷,mamicode.com
首页 > 其他好文 > 详细

bzxoj1090 字符串折叠

时间:2016-02-25 21:10:49      阅读:201      评论:0      收藏:0      [点我收藏+]

标签:

Description

折叠的定义如下: 1. 一个字符串可以看成它自身的折叠。记作S ? S 2. X(S)是X(X>1)个S连接在一起的串的折叠。记作X(S) ? SSSS…S(X个S)。 3. 如果A ? A’, B?B’,则AB ? A’B’ 例如,因为3(A) = AAA, 2(B) = BB,所以3(A)C2(B) ? AAACBB,而2(3(A)C)2(B)?AAACAAACBB 给一个字符串,求它的最短折叠。例如AAAAAAAAAABABABCCD的最短折叠为:9(A)3(AB)CCD。

Input

仅一行,即字符串S,长度保证不超过100。

Output

仅一行,即最短的折叠长度。

区间dp,t[i][j]表示i开头长度为j的串的重复次数,f[i][j]表示i开头长度j的串的最短长度。

#include<cstdio>
#include<cstring>
char s[105];
int f[105][105];
int t[105][105];
int v[105];
inline void mins(int&a,int b){if(a>b)a=b;}
int main(){
    for(int i=1;i<10;i++)v[i]=3;
    for(int i=10;i<100;i++)v[i]=4;
    v[100]=5;
    scanf("%s",s);
    int l=strlen(s);
    for(int i=0;i<=l;i++)
        for(int j=1;j<=l;j++)f[i][j]=j;
    for(int i=1;i<=l;i++){
        t[l-i][i]=1;
        for(int j=i+1;j<=l;j++){
            t[l-j][i]=1;
            bool d=1;
            for(int k=0;k<i;k++){
                if(s[l-j+k]!=s[l-j+i+k]){
                    d=0;
                    break;
                }
            }
            if(d)t[l-j][i]=t[l-j+i][i]+1;
        }
    }
    for(int i=1;i<=l;i++){
        for(int j=2;j<=i;j++){
            if(i%j==0){
                int c=i/j;
                for(int k=0;k<l;k++)
                    if(t[k][c]>=j)mins(f[k][i],f[k][c]+v[j]);
            }
        }
        for(int k=0;k<l;k++)
            for(int a=1;a<i;a++)mins(f[k][i],f[k][a]+f[k+a][i-a]);
    }
    printf("%d",f[0][l]);
    return 0;
}

 

bzxoj1090 字符串折叠

标签:

原文地址:http://www.cnblogs.com/ccz181078/p/5218240.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!