看出来矩阵加速也没看出来KMP……

题目描述

阿申准备报名参加 GT 考试,准考证号为\(N\)位数\(X_1,X_2…X_n(0\le X_i\le9)\),他不希望准考证号上出现不吉利的数字。 他的不吉利数学\(A_1,A_2…A_m(0\le A_i\le 9)\)有\(M\)位,不出现是指\(X_1,X_2…X_n\)中没有恰好一段等于\(A_1,A_2…A_m\)​,\(A_1\)和\(X_1\)可以为\(0\)。

输入输出格式

输入格式:

第一行输入\(N,M,K\),接下来一行输入\(M\)位的数。

输出格式:

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

输入输出样例

输入样例#1:

4 3 100
111
输出样例#1:

81

说明

\(N\le 10^9,M\le 20,K\le 1000\)

题解:

    这个题虽然没怎么提字符串,但是KMP却是解题的关键。

    首先令f[i][j]表示到第i个为止连续匹配了j个的方案数,考虑DP或递推。

    一开始以为所有的转移都不带系数,感觉后面接一个数要么能继续匹配,要么不能继续匹配,只有两种转移情况,然后发现是比较固定的,\(N\le 10^9\)就可以矩阵加速完成推导了。

    不过怎么改都过不了样例。想到后面接的数字有10种情况,只有1种可以连续,就分配1:9的倍数比例转移,当前方案数×9转移到失配,当前方案数×1转移到下一个。这时候样例过了,手敲的几组很弱的数据也没问题。交上去一分没有……

    结果看了一些题解才知道,上面的方案数×9转移到失配中的失配不一定直接到0了,而是可能失配到中间,也就是可以从半路接着匹配到其它的,这里就和KMP不谋而合了。比如

1313匹配到

1312
虽然在最后一位失配了,但是不至于全部重来,可以只用移2位,得到
1313

1312 后面的让之后的转移去做好了

    因此\(f[i][j]=\sum\limits_{k=1}^{m-1}f[i-1][k]*a[i][k][j]\),\(a[i][k][j]\)表示到第i-1个字符连续匹配成功k次转移到第i个字符连续转移j次的方案数。正常情况下\(a[i][k][k+1]=1\),然后KMP的nxt数组就派上用场了,它可以求出下一个字符是c时的最长匹配\(f[i_1][j_1]\),让方案数变为加上\(f[i_1][j_1]\)。

    初始化\(f[0][0]=1\),因为对每一个\(i\),\(a\)数组都是一定的,所以放进矩阵里转移,进行矩阵加速就可以了。状态矩阵是\((m+1)\times 1\)的,转移矩阵就需要\((m+1)\times (m+1)\)。

    就有了$$\begin{bmatrix}
a[0][1] & a[1][1] & \cdots & a[m][1]\\
a[0][2] & a[1][2] & \cdots & a[m][2]\\
\cdots & \cdots & \ddots & \cdots\\
a[m][1] & a[m][2] & \cdots & a[m][m]
\end{bmatrix}
\times \begin{bmatrix}
f[i][0]\\
f[i][1]\\
\cdots \\
f[i][m]
\end{bmatrix}
=\begin{bmatrix}
f[i+1][0]\\
f[i+1][1]\\
\cdots \\
f[i+1][m]
\end{bmatrix}$$

    在计算\(a\)数组时,方法和KMP匹配很像,只不过每次都要匹配'0'~'9',而且不是真正的KMP,所以变量\(j\)都要赋成\(i-1\)。

Code:

#include<cstdio>
#include<cstring>
long long p;
struct matrix//矩阵部分
{
long long a[25][25];
int x,y;
matrix(int x,int y)
{
this->x=x;
this->y=y;
memset(a,0,sizeof(a));
}
matrix()
{
memset(a,0,sizeof(a));
}
matrix mul(matrix m)
{
matrix n(x,m.y);
for(int i=1;i<=x;++i)
for(int j=1;j<=m.y;++j)
for(int k=1;k<=y;++k)
{
n.a[i][j]+=a[i][k]*m.a[k][j]%p;
n.a[i][j]%=p;
}
return n;
}
};
int nxt[30],m;
char t[30];
void kmp()
{
for(int i=2,j=0;i<=m;++i)//单纯求nxt
{
while(j&&t[j+1]!=t[i])
j=nxt[j];
if(t[j+1]==t[i])
++j;
nxt[i]=j;
}
}
int main()
{
int n;
scanf("%d%d%lld",&n,&m,&p);
scanf("%s",t+1);
kmp();
matrix x(m+1,m+1); for(int i=1;i<=m;++i)
for(int j='0';j<='9';++j)//每个都要匹配,位置不一定都相同
{
int k=i-1;
while(k&&(k==m||t[k+1]!=j))
k=nxt[k];
if(t[k+1]==j)
++k;
x.a[k+1][i]++;
}
for(int i=1;i<=m+1;++i)
{
for(int j=1;j<=m+1;++j)
printf("%d ",x.a[i][j]);
puts("");
}
matrix y(m+1,1);
y.a[1][1]=1;
matrix ans(m+1,m+1);
for(int i=1;i<=m+1;++i)
ans.a[i][i]=1;
while(n)
{
if(n&1)
ans=ans.mul(x);
x=x.mul(x);
n>>=1;
}
ans=ans.mul(y);
long long sum=0;
for(int i=1;i<=m;++i)
sum+=ans.a[i][1];
printf("%lld\n",sum%p);
return 0;
}

【KMP】【矩阵加速】【递推】洛谷 P3193 [HNOI2008]GT考试 题解的更多相关文章

  1. 洛谷P3193 [HNOI2008]GT考试(KMP,矩阵)

    传送门 大佬讲的真吼->这里 首先考虑dp,设$f[i][j]$表示长串匹配到第$i$位,短串最多匹配到$j$位时的方案数 那么答案就是$\sum_{i=0}^{m-1}f[n][i]$ 然后考 ...

  2. 洛谷P3193 [HNOI2008]GT考试 kmp+dp

    正解:kmp+dp+矩阵优化 解题报告: 传送门! 啊刚说想做矩阵优化dp的字符串题就找到辣QwQ虽然不是AC自动机的但都差不多嘛QwQ 首先显然可以想到一个dp式?就f[i][j]:凑出i位了,在s ...

  3. 洛谷P3193 [HNOI2008]GT考试(dp 矩阵乘法)

    题意 题目链接 Sol 设\(f[i][j]\)表示枚举到位置串的第i位,当前与未知串的第j位匹配,那么我们只要保证在转移的时候永远不会匹配即可 预处理出已知串的每个位置加上某个字符后能转移到的位置, ...

  4. luogu题解 P1707 【刷题比赛】矩阵加速递推

    题目链接: https://www.luogu.org/problemnew/show/P1707 分析: 洛谷的一道原创题,对于练习矩阵加速递推非常不错. 首先我们看一下递推式: \(a[k+2]= ...

  5. bzoj2004公交线路——DP+矩阵加速递推

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2004 求方案数,想到DP: 因为两个站间距离<=p,所以每p个站中所有车一定都会停靠至 ...

  6. 矩阵经典题目七:Warcraft III 守望者的烦恼(矩阵加速递推)

    https://www.vijos.org/p/1067 非常easy推出递推式f[n] = f[n-1]+f[n-2]+......+f[n-k]. 构造矩阵的方法:构造一个k*k的矩阵.当中右上角 ...

  7. [HDU2294] Pendant - 矩阵加速递推

    Pendant Time Limit: 6000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Sub ...

  8. [模板][题解][Luogu1939]矩阵乘法加速递推(详解)

    题目传送门 题目大意:计算数列a的第n项,其中: \[a[1] = a[2] = a[3] = 1\] \[a[i] = a[i-3] + a[i - 1]\] \[(n ≤ 2 \times 10^ ...

  9. 【csp模拟赛3】bridge.cpp--矩阵加速递推

    题目描述 穿越了森林,前方有一座独木桥,连接着过往和未来(连接着上一题和下一题...). 这座桥无限长. 小 Q 在独木桥上彷徨了.他知道,他只剩下了 N 秒的时间,每一秒的时间里,他会向 左或向右移 ...

随机推荐

  1. 几个常用的HTTP状态码

    200:客户端请求成功 302:临时跳转,跳转的地址通过Location指定 400:客户端请求有语法错误,不能被服务器识别 403:服务器收到请求,但是拒绝提供服务 404:请求的资源不存在 500 ...

  2. [GO]数组的比较和赋值

    package main import "fmt" func main() { //支持比较,只支持==或!=,比较是不是每一个元素都一样,2个数据比较,数据类型要一样 a := ...

  3. (转)关于Update语句的锁

    原文地址:http://www.cnblogs.com/wdfrog/p/3144020.html 环境:MSSQL2005,在Read Committed级别 语句A:begin tranUpdat ...

  4. (转)C# HTML解析示例---星星引发的血案

    原文地址:http://www.cnblogs.com/wurang/archive/2013/06/14/3119023.html [前言] 从CSDN转投cnBlog也有一段时间了,发现cnBlo ...

  5. ntroducing K-Pattern: A Rapid Way to Make CRUD Operations with Entity Framework

    Download Source Introduction Earlier in the last year I made a different approach to work on Entity ...

  6. Python - selenium_WebDriver 鼠标键盘事件

    from selenium import webdriver #引入ActionChains类 提供了鼠标的操作方法 from selenium.webdriver.common.action_cha ...

  7. poj3080 Blue Jeans(暴枚+kmp)

    Description The Genographic Project is a research partnership between IBM and The National Geographi ...

  8. EFCore扩展Update方法(实现 Update User SET Id=Id+1)

    EFCore扩展Update方法(实现 Update User SET Id = Id + 1) 源码地址(github) 前言 EFCore在操作更新的时候往往需要先查询一遍数据,再去更新相应的字段 ...

  9. 十一、Node.js监听代码改动自动重启node插件supervisor

    我们慢慢地发现,每次我们稍微改变js代码都需要重启服务才能在浏览器显示新的效果,很麻烦,这里我们可以通过npm命令安装supervisoror插件,安装方法如下 之前章节我们知道安装了nodejs就会 ...

  10. Go环境搭建(Windows)

    下载MSI MSI地址 配置环境变量 GOPATH: 用于存放Go语言Package的目录,这个目录不能在Go的安装目录中 GOBIN: Go二进制文件存放目录,写成%GOROOT%\bin就好 PA ...