zoukankan      html  css  js  c++  java
  • [BZOJ1009] [HNOI2008]GT考试

    [BZOJ1009] [HNOI2008]GT考试

    Description

    阿申准备报名参加GT考试,准考证号为N位数X1X2....Xn(0<=Xi<=9),他不希望准考证号上出现不吉利的数字。他的不吉利数学A1A2...Am(0<=Ai<=9)有M位,不出现是指X1X2...Xn中没有恰好一段等于A1A2...Am. A1和X1可以为0

    Input

    第一行输入N,M,K.接下来一行输入M位的数。 N<=10^9,M<=20,K<=1000

    Output

    阿申想知道不出现不吉利数字的号码有多少种,输出模K取余的结果.

    Sample Input

    4 3 100
    111

    Sample Output

    81

    试题分析

    由kmp数组构造转移矩阵,然后矩阵快速幂转移。

    #include<iostream>
    #include<cstring>
    #include<cstdio>
    #include<vector>
    #include<algorithm>
     
    using namespace std;
    #define LL long long
     
    inline int read(){
        int x=0,f=1; char c=getchar();
        for(;!isdigit(c);c=getchar()) if(c=='-') f=-1;
        for(;isdigit(c);c=getchar()) x=x*10+c-'0';
        return x*f;
    }
    const int INF = 2147483600;
    const int MAXN = 100010;
     
    int N,M,Mod; int a[MAXN+1]; char str[MAXN+1];
    int nxt[MAXN+1][27];
     
    struct Mat{
        int x,y; int a[101][101];
        inline void init(int rr,int cc){x=rr; y=cc; memset(a,0,sizeof(a));} 
    }A,Ans; int Pw[11]; int sta[101],top;
    Mat operator * (Mat A,Mat B){
        Mat C; C.init(A.x,B.y);
        for(int i=0;i<=A.x;i++){
            for(int j=0;j<=B.y;j++){
                for(int k=0;k<=A.y;k++)
                    C.a[i][j]+=A.a[i][k]*B.a[k][j]%Mod,C.a[i][j]%=Mod;
            }
        } return C;
    }
    Mat operator ^ (Mat A,int P){
        Mat B; B.init(A.x,A.y); for(int i=0;i<=A.x;i++) B.a[i][i]=1;
        for(; P ; P>>=1,A=A*A) if(P&1) B=A*B; return B;
    }  
     
    int main(){
        //freopen(".in","r",stdin);
        //freopen(".out","w",stdout);
        M=read(),N=read(),Mod=read(); scanf("%s",str+1);
        for(int i=1;i<=N;i++) a[i]=str[i]-'0'; 
        for(int i=0;i<10;i++) nxt[0][i]=-1;
        for(int i=1;i<=N;i++){
            for(int c=0;c<10;c++){
                int j=i-1; 
                while(j>-1){
                    if(c==a[nxt[j][a[j]]+1]) {nxt[i][c]=nxt[j][a[j]]+1; break;}
                    else j=nxt[j][a[j]];
                }
            }
        } A.init(N-1,N-1);
        for(int i=0;i<N;i++){
            for(int c=0;c<10;c++){
                if(c==a[i+1]) A.a[i+1][i]++;
                else A.a[nxt[i+1][c]][i]++;
            }
        }
        A=A^M; Ans.init(N-1,0);
        Ans.a[0][0]=1; //Ans.a[0][0]=8; 
        int ans=0;
        Ans=A*Ans; for(int i=0;i<N;i++) (ans+=Ans.a[i][0])%=Mod;
        printf("%d
    ",ans);
        return 0;
    }
    
    
  • 相关阅读:
    2331: [SCOI2011]地板 插头DP
    APIO2018 铜滚记
    2827: 千山鸟飞绝 非旋treap
    3682: Phorni 后缀平衡树 线段树
    4712: 洪水 基于链分治的动态DP
    20180507小测
    4923: [Lydsy1706月赛]K小值查询 平衡树 非旋转Treap
    5312: 冒险 线段树 复杂度分析
    5210: 最大连通子块和 动态DP 树链剖分
    4513: [Sdoi2016]储能表 数位DP
  • 原文地址:https://www.cnblogs.com/wxjor/p/9515772.html
Copyright © 2011-2022 走看看