首页 > 代码库 > 【BZOJ】1090: [SCOI2003]字符串折叠(dp)

【BZOJ】1090: [SCOI2003]字符串折叠(dp)

http://www.lydsy.com/JudgeOnline/problem.php?id=1090

随便yy一下。。

设f[i,j]表示i~j的最小长度

f[i, j]=min{j-i+1, f[i,k]+f[k+1, j], count[x]+2+f[i, i+x-1]},其中count[x]表示x的位数,最后边的转移那个条件是i~j都是长度为x的串连在一起。

然后最后那个转移暴力233可以水过。。。

#include <cstdio>#include <cstring>#include <cmath>#include <string>#include <iostream>#include <algorithm>#include <queue>#include <set>#include <map>using namespace std;typedef long long ll;#define rep(i, n) for(int i=0; i<(n); ++i)#define for1(i,a,n) for(int i=(a);i<=(n);++i)#define for2(i,a,n) for(int i=(a);i<(n);++i)#define for3(i,a,n) for(int i=(a);i>=(n);--i)#define for4(i,a,n) for(int i=(a);i>(n);--i)#define CC(i,a) memset(i,a,sizeof(i))#define read(a) a=getint()#define print(a) printf("%d", a)#define dbg(x) cout << (#x) << " = " << (x) << endl#define error(x) (!(x)?puts("error"):0)#define rdm(x, i) for(int i=ihead[x]; i; i=e[i].next)inline const int getint() { int r=0, k=1; char c=getchar(); for(; c<‘0‘||c>‘9‘; c=getchar()) if(c==‘-‘) k=-1; for(; c>=‘0‘&&c<=‘9‘; c=getchar()) r=r*10+c-‘0‘; return k*r; }const int N=105;char s[N];int f[N][N], c[N];int main() {	scanf("%s", s+1);	for1(i, 1, 100) if(i<10) c[i]=1; else if(i<100) c[i]=2; else c[i]=3;	int n=strlen(s+1);	rep(len, n) for1(i, 1, n) {		int j=i+len;		if(j>n) break;		int &d=f[i][j], l=j-i+1;		d=l;		for1(k, i, j-1) d=min(d, f[i][k]+f[k+1][j]);		for1(x, 1, l) if(l%x==0) {			int flag=1;			for1(now, i, i+x-1) {				int next=now+x;				while(next<=j) { if(s[next]!=s[now]) { flag=0; break; } next+=x; }				if(!flag) break;			}			if(flag) d=min(d, c[l/x]+2+f[i][i+x-1]);		}	}	printf("%d\n", f[1][n]);	return 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

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

Sample Input

NEERCYESYESYESNEERCYESYESYES

Sample Output

14

HINT

 

一个最短的折叠为:2(NEERC3(YES))

 

Source

【BZOJ】1090: [SCOI2003]字符串折叠(dp)