博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
BZOJ 1009 【HNOI2008】 GT考试
阅读量:5332 次
发布时间:2019-06-14

本文共 1417 字,大约阅读时间需要 4 分钟。

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取余的结果.

 
  这道题一眼看去像是一道容斥dp,仔细思考后发现其实普通的dp就可以做了。
  我们令$f_{i,j}$表示准考证号确定了前i位,其中最后一段已经和不吉利串匹配了$j$位的方案数。那么,显然我们只需要枚举每一位选什么数字即可。至于加了一位数字之后最后一段匹配了多少位,完全可以用$kmp$来解决。因为$kmp$算法中$next$数组的含义就是不为整个串的前缀与后缀相等的最大长度。
  但是,这样的复杂度是$O(NM^2)$的。观察发现,$M$特别小,于是可以把转移矩阵预处理出来,使用矩阵快速幂优化即可。
  下面贴代码:
#include
#include
#include
#include
#include
#define File(s) freopen(s".in","r",stdin),freopen(s".out","w",stdout)using namespace std;typedef long long llg;int n,m,k,nt[22],ans;char s[22];void gi(int &x){if(x>=k) x%=k;}struct matrix{ int w[23][23]; matrix(){memset(w,0,sizeof(w));} void fu(){for(int i=0;i
'9'||c<'0')&&c!='-') c=getchar(); if(c=='-') c=getchar(),q=1; while(c>='0'&&c<='9') w=w*10+c-'0',c=getchar(); return q?-w:w;}matrix mi(matrix a,int b){ matrix s; s.fu(); while(b){ if(b&1) s=s*a; a=a*a; b>>=1; } return s;}int main(){ File("a"); n=getint(); m=getint(); k=getint(); scanf("%s",s+1); for(int i=2,j=0;i<=m;i++){ while(j && s[j+1]!=s[i]) j=nt[j]; if(s[j+1]==s[i]) j++; nt[i]=j; } for(int i=0,x;i

转载于:https://www.cnblogs.com/lcf-2000/p/5913233.html

你可能感兴趣的文章
linux的子进程调用exec( )系列函数
查看>>
MSChart的研究
查看>>
C# 索引器
查看>>
MySQLdb & pymsql
查看>>
zju 2744 回文字符 hdu 1544
查看>>
delphi 内嵌汇编例子
查看>>
【luogu P2298 Mzc和男家丁的游戏】 题解
查看>>
前端笔记-bom
查看>>
MATLAB作图方法与技巧(一)
查看>>
上海淮海中路上苹果旗舰店门口欲砸一台IMAC电脑维权
查看>>
Google透露Android Market恶意程序扫描服务
查看>>
给mysql数据库字段值拼接前缀或后缀。 concat()函数
查看>>
迷宫问题
查看>>
【FZSZ2017暑假提高组Day9】猜数游戏(number)
查看>>
泛型子类_属性类型_重写方法类型
查看>>
eclipse-将同一个文件分屏显示
查看>>
mysql5.x升级至mysql5.7后导入之前数据库date出错的解决方法!
查看>>
对闭包的理解
查看>>
练习10-1 使用递归函数计算1到n之和(10 分
查看>>
Oracle MySQL yaSSL 不明细节缓冲区溢出漏洞2
查看>>