方法:区间DP
解析:zxr讲完这道题好几天后我来撸这个。
只记得是什么区间DP?
好,然后我抱着做SB题的内心来做这道题。
这道题是SB题,这道题是SB题,这道题是SB题,恩重要的话说三遍。
f[i][j]表示i到j折叠最小的长度。
转移就是个逗比。
枚举中间点K,
一种是i~k不能成为k+1~j的循环节,那么直接加和
一种是i~k能成为k+1~j的循环节,则可以加和或者折叠,更新最优值。
就那个判断能否成为循环节我调了半天…
太弱了RRRR!
最后还因为循环节个数忘加了本身而被一百个A这种数据卡了!
代码:
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
char s[110];
int f[110][110];
int flag;
void check(int l,int p,int r)
{
if((r-p)%(p-l+1)!=0)return;
int lenth=p-l+1;
for(int i=p+1;i<=r;i++)
{
if(s[i]!=s[(i-l)%lenth+l])return;
}
flag=1;
}
int main()
{
memset(f,0x3f,sizeof(f));
scanf("%s",s+1);
int len=strlen(s+1);
for(int i=1;i<=len;i++)f[i][i]=1;
for(int l=2;l<=len;l++)
{
for(int i=1;i+l-1<=len;i++)
{
int j=i+l-1;
for(int k=i;k<j;k++)
{
f[i][j]=min(f[i][j],f[i][k]+f[k+1][j]);
flag=0;
check(i,k,j);
if(flag)
{
int tmp=f[i][k]+2;
int bit=(j-k)/(k-i+1)+1;
while(bit)bit/=10,tmp++;
f[i][j]=min(f[i][j],tmp);
}
}
}
}
printf("%d\n",f[1][len]);
}