【bzoj1009】【HNOI2008】【GT考试】

1009: [HNOI2008]GT考试

Time Limit: 1 Sec Memory Limit: 162 MB

Submit: 2231 Solved: 1365

[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位的数。 100%数据N<=10^9,M<=20,K<=1000 40%数据N<=1000 10%数据N<=6

Output

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

Sample Input

4 3 100

111

Sample Output

81

设f[i][j]表示长度为i的字符串后j位于前j为匹配的方案数。

那么怎样转移呢?

设a[x][y]为f[i-1][x]转移到f[i][y]的方案数。

那么a[x][y]就表示长度为x的前缀上加上一个数字后,后缀的最长长度为y的前缀匹配,可以加的数字有多少种。

那么这个a数组我们可以用kmp求出来,然后再矩乘一下就行了。

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int n,m,p,next[30],a[30][30],ans[30][30],c[30][30];
char s[30];
void kmp()
{
    int i,f=0;
    char j;
    for(i=2;i<=m;i++){
        while(f&&s[f+1]!=s[i]) f=next[f];
        if(s[f+1]==s[i]) f+=1;
        next[i]=f;
    }
    for(i=0;i<m;i++)
      for(j=‘0‘;j<=‘9‘;j++){
        f=i;
        while(f&&s[f+1]!=j) f=next[f];
        if(j==s[f+1]) a[i][f+1]+=1;
        else a[i][0]+=1;
      }
}
int main()
{
    int i,j,sum=0,k;
    scanf("%d%d%d%*c%s",&n,&m,&p,s+1);
    kmp();
    for(i=0;i<m;++i) ans[i][i]=1;
    while(n){
        if(n&1){
            for(i=0;i<m;++i)
              for(j=0;j<m;++j){
                c[i][j]=0;
                for(k=0;k<m;++k)
                  c[i][j]=(c[i][j]+a[i][k]*ans[k][j])%p;
              }
            for(i=0;i<m;++i)
              for(j=0;j<m;++j)
                ans[i][j]=c[i][j];
        }
        n>>=1;
        for(i=0;i<m;++i)
          for(j=0;j<m;++j){
            c[i][j]=0;
            for(k=0;k<m;++k)
              c[i][j]=(c[i][j]+a[i][k]*a[k][j])%p;
          }
        for(i=0;i<m;++i)
          for(j=0;j<m;++j)
            a[i][j]=c[i][j];
    }
    for(i=0;i<m;++i)
      sum=(sum+ans[0][i])%p;
    printf("%d\n",sum);
}

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-08-02 09:36:46

【bzoj1009】【HNOI2008】【GT考试】的相关文章

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位的数. 100%数据N<=10^9,M<=20,K<=1000 40%数据N<

bzoj1009 [HNOI2008] GT考试 矩阵乘法+dp+kmp

1009: [HNOI2008]GT考试 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 4542  Solved: 2815[Submit][Status][Discuss] Description 阿申准备报名参加GT考试,准考证号为N位数X1X2....Xn(0<=Xi<=9),他不希望准考证号上出现不吉利的数字.他的不吉利数学A1A2...Am(0<=Ai<=9)有M位,不出现是指X1X2...Xn中没有恰好一段等于A1A2..

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位的数. 100%数据N<=10^9,M<=20,K<=1000 40%数据N<=1000 10%数据N<=6 O

[BZOJ1009] [HNOI2008] GT考试 (KMP &amp; dp &amp; 矩阵乘法)

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取余的

BZOJ1009: [HNOI2008]GT考试 矩阵快速幂+kmp+dp

这个题你发现打暴力的话可以记忆化搜素加剪枝,那么意味着可以递推,我们搜的话就是1010^9我们就往下匹配遇到匹配成功就return,那么我们可以想一下什么决定了状态,我们考虑kmp的过程,对于我们目前匹配到的距离,下一次在匹配时不会用他之后的字符,那么只要我们知道匹配到的距离和已匹配长度就行了,那么我们考虑状态的转移,我们由于要像kmp那样匹配于是我们只要知道在匹配到k位时往下走一个数时匹配到哪,算出a[k][j](在k时到j的方案数),那么新的f[i][j]=∑f[i-1][k]*a[k][j

【以前的空间】bzoj1009 [HNOI2008]GT考试

动态规划+kmp+矩阵快速幂 关于这题可以写出一个dp方程(f[i,j]表示准考证前i位中后j位为不吉利的数字的前j位的情况的个数) f[i,j]=Σf[i-1,k],其中j表示不吉利数字前k个数字加上某个数字后变成为不吉利数字的前j位(比如不吉利数字122123,然后现在k=5,那么如果填个3,j=6(123123):填个2,j=3(122):填个1,j=1(1):填个0,j=0. 然后我们就可以发现--好像可以用kmp算法来优化每次k+某个数字可以转移到的j的位置--因为j包括了前k个数字,

BZOJ1009 HNOI2008 GT考试 一般DP+矩阵乘法+KMP

题意:给定一个长度为M的字符串A,求长度为N的字符串中,子串中不包含A的字符串的数量,其中字符串仅由‘0’-‘9’组成. 题解:设f[i][j]=长度为i最后几位能匹配A的前j个字符的字符串种数,那么每往后添加一个字符,能转移到的位置通过KMP的Next数组很轻松就能找到.那么我们就能构造出来一个矩阵,a[i][j]=1表示可以通过在A[i]后面加一个字符,使得A[1]-A[j]成为原有字符串的子串.快速幂优化后答案就是f[N][0]+……+f[N][M-1] #include <cstdio>

[BZOJ1009][HNOI2008]GT考试(KMP+DP)

[不稳定的传送门 Solution dp[i][j]表示前i个字符当前匹配到不吉利串的第j个,即当前方案的后缀等于不吉利串前缀 然而由于n过大,不能直接转移,用矩阵优化 Code #include <cstdio> #include <algorithm> #include <cstring> using namespace std; char s[120]; int n,m,mo,nex[120]; inline int read(){ int x=0,f=1;cha

【BZOJ1009】[HNOI2008]GT考试 next数组+矩阵乘法

[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 阿申想知

BZOJ 1009: [HNOI2008]GT考试 AC自动机+矩阵快速幂

经典题目了....虽然只有一个不能出现的字符串,但还是写了ac自动机 1009: [HNOI2008]GT考试 Time Limit: 1 Sec  Memory Limit: 162 MB Submit: 2051  Solved: 1257 [Submit][Status][Discuss] Description 阿申准备报名参加GT考试,准考证号为N位数X1X2....Xn(0<=Xi<=9),他不希望准考证号上出现不吉利的数字.他的不吉利数学A1A2...Am(0<=Ai<