bzoj1090: [SCOI2003]字符串折叠 区间dp+hash

8 篇文章 0 订阅
2 篇文章 0 订阅

题目大意:一个重复k次的子串x可以折叠替换成k(x)的形式,问原串经折叠后的最短长度. n100 .
k(x)的长度为 k的位数+x的长度+2.
区间dp,转移方程为 f[i][j]=min(ji+1,f[i][k]+f[k+1][j])
如果i~j的子串可以由i~i+k-1的子串重复得来
就要再转移 f[i][j]=min(f[i][j],f[i][i+k1]+d[len/k]+2)
d[i]表示数字i的长度.
用hash判断两个子串是否相同。
另外这里判断的时候,不需要判断len/k次,
只需要判i~j-k和j-k+1~j是否相同即可。

#include <bits/stdc++.h>
#define ull unsigned long long
#define clr(x,i) memset(x,i,sizeof(x)) 
using namespace std;
const int N=105;
const ull seed=131;
char str[N];
int n,a[N],d[N],f[N][N];
ull h[N],pw[N];

void solve()
{
    clr(f,60);
    for(int i=1;i<=n;i++)
      f[i][i]=1;
    for(int len=2;len<=n;len++)
      for(int i=1;i+len-1<=n;i++)
      {
        int j=i+len-1;
        for(int k=i;k<j;k++)
          f[i][j]=min(f[i][j],f[i][k]+f[k+1][j]);
        for(int k=1;i+k-1<=j;k++)
          if(len%k==0)
          {
            ull h1=h[j-k]-h[i-1]*pw[len-k];//i~(j-k)
            ull h2=h[j]-h[i+k-1]*pw[len-k];//(i+k-1)~j
            if(h1==h2)
              f[i][j]=min(f[i][j],f[i][i+k-1]+d[len/k]+2);
          }
      }
    printf("%d\n",f[1][n]);
}
void init()
{
    pw[0]=1;
    for(int i=1;i<=n;i++)pw[i]=pw[i-1]*seed;
    for(int i=1;i<=n;i++)
      a[i]=str[i]-'a'+1;
    for(int i=1;i<=n;i++)
      h[i]=h[i-1]*seed+a[i];
    for(int i=1;i<=n;i++)
      d[i]=(i/10)+1;
}
int main()
{
    scanf("%s",str+1);n=strlen(str+1);
    init();
    solve();
    return 0;
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值