zoukankan      html  css  js  c++  java
  • BZOJ1009

    原题链接

    Description

    给出一个长度为m(m20)的数字串,求不包含这个数字串的n(n109)位数字串有多少个。答案对k(k103)取模。

    Solution

    矩阵乘法优化DP。
    f[i][j]表示末尾至少再补上j个数字才能形成模式串的i位数字串的个数。举例来说,模式串是12345,则f[i][1]表示以1234结尾的i位数字串的个数(再补一个数5),f[i][2]表示以123结尾的i位数字串的个数(再补两个数45)…
    注意至少!比如对于模式串12123,结尾是1212的串应该只属于f[i][1]而不属于f[i][3]

    那么,如何构造由[f[i][1]...f[i][m]]转移为[f[i+1][1]...f[i+1][m]]的矩阵呢?
    i+1位数字串是由i位数字串在末尾加上一位数字得到的,所以我们可以通过枚举这一位上的数字来算出这个矩阵。再次举例,模式串121的转移矩阵为:

    010011989
    A11=0表示以12结尾的串,无论在末尾加上什么数也不能变成以12结尾的串。
    A12=0表示以12结尾的串,无论在末尾加上什么数也不能变成以1结尾的串。
    A13=9表示以12结尾的串,在末尾加上023456789这九个数能够变成不以112结尾的串。
    A21=1表示以1结尾的串,在末尾加上2这个数能够变成以12结尾的串。
    A22=1表示以1结尾的串,在末尾加上1这个数能够变成以1结尾的串。
    A23=8表示以1结尾的串,在末尾加上03456789这八个数能够变成不以112结尾的串。

    可以看到第一行元素和是9,其他行元素和是10。因为差1位就形成模式串的话,补上那位就不合法啦。
    初始状态[00...1],再乘以转移矩阵Tn次方,答案就是最终得到的矩阵各元素之和。

    时间复杂度O(m3logn)

    Code

    //[HNOI2008]GT考试
    #include <cstdio>
    #include <cstring>
    int n,m,P; char s[30],s0[30];
    struct matrix
    {
        int row,col,v[30][30];
        matrix operator *(matrix b)
        {
            int p=row,q=col,r=b.col;
            matrix c; c.row=p,c.col=r;
            for(int i=1;i<=p;i++)
                for(int j=1;j<=r;j++)
                {
                    c.v[i][j]=0;
                    for(int k=1;k<=q;k++)
                        c.v[i][j]+=v[i][k]*b.v[k][j],c.v[i][j]%=P;
                }
            return c;
        }
    }x,y;
    int pre(int L)
    {
        for(int i=1;i<=L;i++)
        {
            int j=i;
            while(s[j-i+1]==s0[j]&&j<=L) j++;
            if(j==L+1) return m-(L-i+1); 
        }
        return m;
    }
    matrix pow(matrix a,int b)
    {
        matrix res; res.row=res.col=a.row;
        for(int i=1;i<=res.row;i++)
            for(int j=1;j<=res.col;j++)
                res.v[i][j]=(i==j);
        matrix t=a;
        while(b>0)
        {
            if(b&1) res=res*t;
            t=t*t; b>>=1;
        }
        return res;
    }
    int main()
    {
        scanf("%d%d%d",&n,&m,&P);
        scanf("%s",s+1);
        x.row=x.col=m;
        for(int i=1;i<=m;i++)
        {
            memset(s0,0,sizeof s0);
            for(int j=1;j<=m-i;j++) s0[j]=s[j];
            for(char j='0';j<='9';j++) s0[m-i+1]=j,x.v[i][pre(m-i+1)]++;
        }
        x.v[1][0]=0;
        x=pow(x,n);
        y.row=1,y.col=m,y.v[1][m]=1; y=y*x;
        int ans=0;
        for(int i=1;i<=m;i++) ans+=y.v[1][i];
        printf("%d
    ",ans%P);
        return 0;
    }

    P.S.

    再这样咸鱼…会死…

  • 相关阅读:
    Newbe.Claptrap 框架入门,第三步 —— 定义 Claptrap,管理商品库存
    Hudi on Flink在顺丰的实践应用
    Kafka监控必备——Kafka-Eagle 2.0.2正式发布
    如何将炫酷的报表直接截图发送邮件——在Superset 0.37使用Schedule Email功能
    突发!美商务部宣布封禁微信,TikTok——面对科技封锁,如何应对
    离线安装Superset 0.37
    Windows系统快速安装Superset 0.37
    Superset 0.37 发布——颜值最高的数据可视化平台
    Hive查看,删除分区
    超详细,Windows系统搭建Flink官方练习环境
  • 原文地址:https://www.cnblogs.com/VisJiao/p/8485742.html
Copyright © 2011-2022 走看看