zoukankan      html  css  js  c++  java
  • BZOJ3670: [Noi2014]动物园

    【传送门:BZOJ3670


    简要题意:

      给出一个st字符串,定义一个sum数组,sum[i]表示st[1...i]中,所有前缀和后缀不重叠的情况下,前缀和后缀相等的数量,比如说st='aaaaa',sum[5]=2,因为st[1]=st[len],st[1...2]=st[len-1...len],所以有两个,求出(sum[1]+1)*(sum[2]+1)*(sum[3]+1)*...*(sum[n]+1),答案要%10^9+7


    题解:

      原题就告诉了你KMP这个算法,就用KMP来做这道题(仅此题,别的不一定,比如说某省选的一道题上说,考整体二分套可持久化平衡树,结果不是)

      首先像平时求kmp一样求p数组,然后定义一个dep数组,dep[i]表示p[i]是经过多少次向前继承得到的(有点绕口,但是很重要!要理解好),并且我们设如果p[i]=0时,dep[i]=1,比如说st='abcababc',p数组=0 0 0 1 2 1 2 3,dep数组=1 1 1 2 2 2 2 2,求出dep有什么用呢?

      dep就是sum数组的雏形,它表示所有前缀和后缀相等的数量,有可能有重叠的前缀和后缀,也有可能就是整个子串,所以为什么p[i]=0时,dep[i]=1

      接下来就是要得出正确的dep数组,其实怎么来求呢,操作其实和求p数组的操作相似,不过一开始要先将j变为长度小于等于i/2,因为不能重叠,然后ans直接在每一次操作中操作


    参考代码:

    #include<cstdio>
    #include<cstring>
    #include<cstdlib>
    #include<algorithm>
    #include<cmath>
    using namespace std;
    typedef long long LL;
    char st[1100000];
    LL dep[1100000];
    LL p[1100000];
    LL sum[1100000];
    int main()
    {
        int T;
        scanf("%d",&T);
        while(T--)
        {
            scanf("%s",st+1);
            int len=strlen(st+1);
            LL ans=1;
            memset(p,0,sizeof(p));
            memset(sum,0,sizeof(sum));
            memset(dep,0,sizeof(dep));
            dep[1]=1;
            for(int i=2;i<=len;i++)
            {
                int j=p[i-1];
                while(st[j+1]!=st[i]&&j!=0) j=p[j];
                if(st[j+1]==st[i]) j++;
                p[i]=j;
                dep[i]=dep[j]+1;
            }
            int j=0;
            for(int i=2;i<=len;i++)
            {
                while((j+1)*2>i) j=p[j];
                while(j!=0&&st[i]!=st[j+1]) j=p[j];
                if(st[i]==st[j+1]) j++;
                ans=(ans*(dep[j]+1))%1000000007;
            }
            printf("%lld
    ",ans);
        }
        return 0;
    }

     

  • 相关阅读:
    期中考试补完计划
    c++简单程序设计-4
    c++简单程序设计-3
    c++简单程序设计-2
    c++简单程序设计-1
    layer ifram 弹出框
    定时发邮件
    二维码生成
    文件或文件夹的压缩
    excel 导入 与 导出
  • 原文地址:https://www.cnblogs.com/Never-mind/p/7865849.html
Copyright © 2011-2022 走看看