【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

你可能感兴趣的:(2003)