zoukankan      html  css  js  c++  java
  • 【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

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

    Sample Input

    4 3 100
    111

    Sample Output

    81

    题解:虽然AC自动机的fail和KMP的next只差了那么一点点,但为什么感觉AC自动机比KMP好理解100倍~

    好吧我们还是用KMP,先求出next数组,然后用f[i][j]表示i位数,第i位数匹配到了模板串的第j位时的方案数

    然后从0..9枚举第i+1位,设加入了第i+1位后匹配到了位置k,则有f[i+1][k]+=f[i][j]

    若第i位正好匹配成功,此时k=m,则f[i+1][m]+=f[i][j]

    显然我们可以用矩乘来优化这个DP过程,我们令x[i][j]表示经过1次匹配后,从位置i匹配到了位置j的方案数。那么对于上面所有符合条件的(j,k),我们都令x[j][k]=1,初始ans[0][0]=1。然后ans*=x^N,答案就是∑ans[0][0...m-1]

    #include <cstdio>
    #include <cstring>
    #include <iostream>
    using namespace std;
    int next[25];
    char str[25];
    typedef struct matrix
    {
    	int v[25][25];
    }M;
    M x,ans,emp;
    int n,m,mod,sum;
    M mmul(M a,M b)
    {
    	M c=emp;
    	int i,j,k;
    	for(i=0;i<=m;i++)
    		for(j=0;j<=m;j++)
    			for(k=0;k<=m;k++)
    				c.v[i][j]=(c.v[i][j]+a.v[i][k]*b.v[k][j])%mod;
    	return c;
    }
    void pm(int y)
    {
    	while(y)
    	{
    		if(y&1)	ans=mmul(ans,x);
    		x=mmul(x,x),y>>=1;
    	}
    }
    int main()
    {
    	scanf("%d%d%d%s",&n,&m,&mod,str);
    	int i=0,j=-1,k;
    	next[0]=-1;
    	while(i<m-1)
    	{
    		if(j==-1||str[i]==str[j])	next[++i]=++j;
    		else	j=next[j];
    	}
    	for(i=0;i<m;i++)
    	{
    		for(j=0;j<=9;j++)
    		{
    			k=i;
    			while(k!=-1&&str[k]-'0'!=j)	k=next[k];
    			x.v[i][k+1]++;
    		}
    	}
    	x.v[m][m]=10,ans.v[0][0]=1;
    	pm(n);
    	for(i=0;i<m;i++)	sum=(sum+ans.v[0][i])%mod;
    	printf("%d",sum);
    	return 0;
    }
  • 相关阅读:
    flexible.js 移动端自适应方案
    Vue为什么不能检测数组变动
    Vue 组件间通信六种方式
    训练首个神经网络:基本分类
    对seq2seq的粗浅认识
    数学模型的过拟合和欠拟合
    在二叉树中寻找值最大的节点并返回——LintCode入门
    Android 包管理机制
    自定义View的三种实现方式及自定义属性使用介绍
    Paint.setFlags中flag意义及使用方法
  • 原文地址:https://www.cnblogs.com/CQzhangyu/p/6791793.html
Copyright © 2011-2022 走看看