[BZOJ1009][HNOI2008]GT考试

1009: [HNOI2008]GT考试

Time Limit: 1 Sec  Memory Limit: 162 MB Submit: 4024  Solved: 2452 [Submit][Status][Discuss]

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
 
设$dp[i][j]$表示前i个字符,且后缀为不吉利数字的前缀的最长长度为j的方案数,那么显然$ans = \sum_{i=1}^{m-1}\ dp[n][i]$
设$num[i][j]$为在长度为$i$的不吉利数字的前缀后加一个数使得后缀为不吉利数字的前缀的最长长度为$j$的方案数,$num$数组可以用kmp求出来
转移为$dp[k][j] = \sum_{i = 1}^{m - 1}\ dp[k -1][i] * num[i][j]$
对于同一串不吉利数字$num$数组为常数,那么可以利用矩阵加速递推
 
#include <cstdio>
#include <cstring> 
const int maxm = 25;
int mod;
struct matrix{
    int n, m, num[maxm][maxm];
    matrix(){}
    matrix(int _n, int _m){
        n = _n;
        m = _m;
        memset(num, 0, sizeof num);
    }
    matrix operator * (const matrix &a){
        matrix b(n, a.m);
        for(int i = 0; i < n; i++)
            for(int j = 0; j < a.m; j++)
                for(int k = 0; k < m; k++)
                    (b.num[i][j] += num[i][k] * a.num[k][j]) %= mod;
        return b;
    }
};
matrix ksm(matrix a, int b){
    matrix s(a.m, a.m);
    for(int i = 0; i < s.m; i++)
        s.num[i][i] = 1;
    while(b){
        if(b & 1) s = s * a;
        b >>= 1;
        a = a * a;
    }
    return s;
}
int n, m;
char A[maxm];
int p[maxm];
int main(){
    scanf("%d %d %d", &n, &m, &mod);
    scanf("%s", A + 1);
    p[1] = 0;
    for(int j = 0, i = 2; i <= m; i++){
        while(A[i] != A[j + 1] && j) j = p[j];
        if(A[i] == A[j + 1]) j++;
        p[i] = j;
    }
    matrix ans(1, m), zy(m, m);
    for(int i = 0; i < m; i++)
        for(int j = '0'; j <= '9'; j++){
            int k = i;
            while(k && A[k + 1] != j) k = p[k];
            if(A[k + 1] == j) k++;
            if(k != m) zy.num[i][k]++;
        }
    ans.num[0][0] = 1;
    ans = ans * ksm(zy, n);
    int aa = 0;
    for(int i = 0; i < m; i++)
        (aa += ans.num[0][i]) %= mod;
    printf("%d\n", aa);
    return 0;
}

 

posted @ 2017-08-27 15:03  jzyy  阅读(133)  评论(0编辑  收藏  举报