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<=1000 10%数据N<=6

Output

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

Sample Input

4 3 100
111

Sample Output

81

题解

  设dp[i][j]表示总共到做第i位,匹配到第j位的ans,那么对于dp[i][j]到dp[i+1][j‘],一定是有固定的转移试的,不会因为运算的值改变而改变。那么想到了什么?矩阵乘法!!设p[i][j]表示匹配到第i位之后,再匹配j,所能得到的新的匹配长度,则可以构建矩阵:对于每一个p[i][j],矩阵的第p[i][j]行第i列加1。

  至于匹配p[i][j],两种方法-->暴力匹配或者kmp都可以。

  矩阵匹配长度范围是:0~m-1。注意,匹配不可到达m,因为如果匹配到了m,那么就是一个不吉利的串出现了!所以只能匹配到第m-1个。那么在kmp的时候,对于匹配0个的转移要特殊处理。

还是举一个例子吧——

  对于111的转移:

  要转移到匹配0位,那么如果之前匹配了0或者1或者2位,只要再接下去一个非1的数字即可转移到,那么:

  dp[i+1][0]= 9* dp[i][0] + 9* dp[i][1] + 9* dp[i][2]

  同理:

  dp[i+1][1]= 1* dp[i][0] + 0* dp[i][1] + 0* dp[i][2]

  dp[i+1][2]= 0* dp[i][0] + 1* dp[i][1] + 0* dp[i][2]

  再来一个稍微复杂一些的:1213

      dp[i][0] dp[i][1] dp[i][2] dp[i][3]

dp[i+1][0]=  9*   8*   9*   9*

dp[i+1][1]=  1*   1*   0*   1*

dp[i+1][2]=  0*   1*   0*   0*

dp[i+1][3]=  0*   0*   1*   0*

所以,根据dp方程就可以构建矩阵了,然后跑矩阵快速幂,就可以拿到满分了!

代码

#include <cstring>
#include <algorithm>
#include <cstdlib>
#include <cstdio>
#include <cmath>
using namespace std;
const int M=20+5;
int n,m,mod;
struct Mat{
    int v[M][M];
    void set(int x){
        memset(v,0,sizeof v);
        if (x==1)
            for (int i=0;i<m;i++)
                v[i][i]=1;
    }
    Mat operator * (Mat x){
        Mat ans;
        ans.set(0);
        for (int i=0;i<m;i++)
            for (int j=0;j<m;j++)
                for (int k=0;k<m;k++){
                    ans.v[i][j]+=v[i][k]*x.v[k][j];
                    if (ans.v[i][j]>=mod)
                        ans.v[i][j]%=mod;
                }
        return ans;
    }
}M1,My,Mans;
Mat Pow(int y){
    if (y==0)
        return M1;
    Mat x=Pow(y/2);
    x=x*x;
    if (y&1)
        x=x*My;
    return x;
}
int p[M][12],next[M];
char ch[M];
int main(){
    scanf("%d%d%d%s",&n,&m,&mod,&ch);
    int k=0;
    memset(next,0,sizeof next);
    for (int i=1;i<m;i++){
        while (k>0&&ch[i]!=ch[k])
            k=next[k-1];
        if (ch[i]==ch[k])
            k++;
        next[i]=k;
    }
    for (int j=0;j<=9;j++)
        if (ch[0]==j+‘0‘)
            p[0][j]=1;
        else
            p[0][j]=0;
    for (int i=1;i<m;i++)
        for (int j=0;j<=9;j++){
            char chj=j+‘0‘;
            int k=i;
            while (k>0&&ch[k]!=chj)
                k=next[k-1];
            if (ch[k]==chj)
                k++;
            p[i][j]=k;
        }
    M1.set(1);
    My.set(0);
    for (int i=0;i<m;i++)
        for (int j=0;j<=9;j++)
            My.v[i][p[i][j]]++;
    Mans=Pow(n);
    int ans=0;
    for (int i=0;i<m;i++)
        ans=(ans+Mans.v[0][i])%mod;
    printf("%d",ans);
    return 0;
}
时间: 2024-11-05 15:58:55

BZOJ1009 [HNOI2008]GT考试 矩阵的相关文章

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考试 矩阵快速幂+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 &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考试 一般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考试

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+矩阵快速幂 关于这题可以写出一个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考试(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考试_(动态规划+kmp+矩阵乘法优化+快速幂)

描述 http://www.lydsy.com/JudgeOnline/problem.php?id=1009 字符串全部由0~9组成,给出一个串s,求一个长度为n的串,不包含s的种类有多少. 分析 第一眼以为是组合.然后更滑稽的是用错误的方法手算样例居然算出来是对的...我数学是有多差... 题解也是看了好半天,有点难理解. 感觉PoPoQQQ神犇讲得还是比较清楚的.传送门:http://blog.csdn.net/popoqqq/article/details/40188173 我们用dp[