题目链接:

https://www.lydsy.com/JudgeOnline/problem.php?id=1009

题意:

准考证号为\(n\)位数\(X_1X_2....X_n(0<=X_i<=9)\),你不希望准考证号上出现不吉利的数字。

不吉利数字\(A_1A_2...A_m(0<=A_i<=9)\)有\(m\)位,不出现是指\(X_1X_2...X_n\) 中没有恰好一段等于\(A_1A_2...A_m\)。\(A_1\) 和 \(X_1\) 都可以为\(0\)。

问你不出现不吉利数字的号码有多少种,输出模 \(mod\) 取余的结果。

\(n<=10^9,M<=20,mod<=1000\)。

题解:

对于这种间接的多模式匹配字符串+计数问题,我们可以用\(AC\)自动机 + \(dp\)。

\(dp[i][j]\)表示串长为 \(i\) , 匹配到了 \(AC\) 自动机的节点 \(j\),并且从来没有出现过完整的不吉利号码的方案数。

容易实现,\(dp\) 形式就是这样:

  dp[0][0] = 1;
for(int i = 1; i <= n; i++) {
for(int j = 0; j <= sz; j++) {
for(int k = 0; k <= 9; k++) {
int v = ch[j][k];
if(End[v] == 0) {
dp[i][v] += dp[i - 1][j];
dp[i][v] %= mod;
}
}
}
}
int res = 0;
for(int i = 0; i <= sz; i++) {
res += dp[n][i];
res %= mod;
}
std::cout << "res = " << res << '\n';

但是\(n<=10^9\),时间复杂度会爆炸。

所以可以用矩阵快速幂来优化加速\(dp\)。

初始化矩阵\(tmp = [1,0,0]\),意思和 \(dp[0][0] = 1\) 的初始化一样。

将 \(x\) 节点与 \(x\) 的所有儿子节点 \(son\) 在转移矩阵 \(a\) 中都分别 \(a[x][son]+=1\)。

这样在矩阵乘法的转移时,可以把父亲的状态转移到儿子那里去。

最后答案即为 \(tmp * a ^ n\)。

建立失配指针时,需要注意:

如果一个数字的后缀是不吉利的(即fail指针指向不吉利数字的结尾节点), 那么这个数字一定也是不吉利的。

代码:

#include <bits/stdc++.h>
using namespace std;
const int maxn = 1000100;
int mod; // AC自动机有三个部分,分别是建树,获取失配指针和查询。
// Aho_Corasick_Automaton :可以简单的理解为将KMP放在Trie树上 // https://www.lydsy.com/JudgeOnline/problem.php?id=1009 int End[maxn];
int ch[maxn][26];
int fail[maxn];
int sz; struct matrix
{
int m[26][26];
void init() {
memset(m,0,sizeof(m));
}
};
matrix operator *(matrix a, matrix b)
{
matrix ans;
ans.init();
for(int i = 1; i <= sz + 1; i++) {
for(int j = 1; j <= sz + 1; j ++) {
for(int k = 1; k <= sz + 1; k++) {
ans.m[i][j] += a.m[i][k] * b.m[k][j];
ans.m[i][j] %= mod;
}
}
}
return ans;
}
matrix qpower(matrix a, int b)
{
matrix ans;
ans.init();
for(int i = 1; i <= sz + 1; i++) {
ans.m[i][i] = 1;
}
while(b > 0)
{
if(b & 1) ans = ans * a;
b >>= 1;
a = a * a;
}
return ans;
}
void insert(string s)
{
int now = 0;
for(int i = 0; i < (int)s.size();i++) {
int c = s[i] - '0';
if(!ch[now][c])ch[now][c] = ++sz;
now = ch[now][c];
}
End[now] = 1;
}
void getfail()
{
queue<int>que;
for(int i = 0; i <= 9; i++) {
if(ch[0][i]) {
que.push(ch[0][i]);
fail[ch[0][i]] = 0;
}
}
while(!que.empty())
{
int u = que.front();
que.pop();
for(int i = 0; i <= 9; i++) {
int v = ch[u][i];
if(v)
{
fail[v] = ch[fail[u]][i];
// 注意如果一个数字的后缀是不吉利的(即fail指针指向不吉利数字的结尾节点)
// 那么这个数字一定也是不吉利的
End[v] |= End[ch[fail[u]][i]];
que.push(v);
}
else ch[u][i] = ch[fail[u]][i];
}
}
}
string s;
int n,m;
int dp[maxn][26];
// dp[i][j]表示串长为 i , 匹配到了 AC 自动机的节点 j,并且从来没有出现过完整的不吉利号码的方案数。
int main(int argc, char const *argv[]) {
// freopen("in.txt","r",stdin);
std::cin >> n >> m >> mod;
std::cin >> s;
insert(s);
getfail(); // dp[0][0] = 1;
// for(int i = 1; i <= n; i++) {
// for(int j = 0; j <= sz; j++) {
// for(int k = 0; k <= 9; k++) {
// int v = ch[j][k];
// if(End[v] == 0) {
// dp[i][v] += dp[i - 1][j];
// dp[i][v] %= mod;
// }
// }
// }
// }
// int res = 0;
// for(int i = 0; i <= sz; i++) {
// res += dp[n][i];
// res %= mod;
// }
// std::cout << "res = " << res << '\n'; matrix a,tmp;
a.init();
tmp.init();
tmp.m[1][1] = 1;
for(int i = 0; i <= sz; i++) {
for(int j = 0; j <= 9; j++) {
int v = ch[i][j];
if(End[v] == 0) {
a.m[i + 1][v + 1] += 1;
}
}
}
a = tmp * qpower(a,n); int ans = 0;
for(int i = 1; i <= sz + 1; i++) {
ans += a.m[1][i];
ans %= mod;
}
std::cout << ans << '\n';
return 0;
}

BZOJ 1009 GT考试 (AC自动机 + 矩阵乘法加速dp)的更多相关文章

  1. [BZOJ 1009] [HNOI2008] GT考试 【AC自动机 + 矩阵乘法优化DP】

    题目链接:BZOJ - 1009 题目分析 题目要求求出不包含给定字符串的长度为 n 的字符串的数量. 既然这样,应该就是 KMP + DP ,用 f[i][j] 表示长度为 i ,匹配到模式串第 j ...

  2. bzoj 2553: [BeiJing2011]禁忌 AC自动机+矩阵乘法

    题目大意: http://www.lydsy.com/JudgeOnline/problem.php?id=2553 题解: 利用AC自动机的dp求出所有的转移 然后将所有的转移储存到矩阵中,进行矩阵 ...

  3. P3193 [HNOI2008]GT考试(KMP+矩阵乘法加速dp)

    P3193 [HNOI2008]GT考试 思路: 设\(dp(i,j)\)为\(N\)位数从高到低第\(i\)位时,不吉利数字在第\(j\)位时的情况总数,那么转移方程就为: \[dp(i,j)=dp ...

  4. 【bzoj1444】[Jsoi2009]有趣的游戏 AC自动机+矩阵乘法

    题目描述 输入 注意 是0<=P 输出 样例输入 样例输出 题解 AC自动机+矩阵乘法 先将所有字符串放到AC自动机中,求出Trie图. 然后构建邻接矩阵:如果x不是某个字符串的末位置,则x连向 ...

  5. bzoj 1444 AC自动机 + 矩阵乘法 | 高斯消元

    恶补了一下AC自动机,花了一天时间终于全部搞明白了. 思路:将每个人的串加入AC自动机,在AC自动机生成的状态图上建边,注意单词末尾的节点只能转移到自己概率为1, 然后将矩阵自乘几十次后误差就很小了, ...

  6. 【POJ2778】AC自动机+矩阵乘法

    DNA Sequence Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 14758 Accepted: 5716 Descrip ...

  7. POJ 2778 DNA Sequence (AC自动机,矩阵乘法)

    题意:给定n个不能出现的模式串,给定一个长度m,要求长度为m的合法串有多少种. 思路:用AC自动机,利用AC自动机上的节点做矩阵乘法. #include<iostream> #includ ...

  8. 洛谷 P4569 - [BJWC2011]禁忌(AC 自动机+矩阵乘法)

    题面传送门 又好久没做过 AC 自动机的题了,做道练练手罢( 首先考虑对于某个固定的字符串怎样求出它的伤害,我们考虑贪心,每碰到出现一个模式串就将其划分为一段,最终该字符串的代价就是划分的次数.具体来 ...

  9. 【poj2778-DNA Sequence】AC自动机+矩阵乘法

    题意: (只含AGCT)给定m个病毒串,让你构造一个长度为n的字符串(也只含有AGCT),问有多少种方案.n很大:1<=n<=2000000000 题解: 用病毒串建立AC自动机(num个 ...

随机推荐

  1. 协变 & 逆变

    都跟里氏替换原则有关. 协变:你可以用一个子类对象去替换相应的一个父类对象,这是完全符合里氏替换原则的,和协(谐)的变.如:用Swan替换Bird. 逆变:你可以用一个父类对象去替换相应的一个子类对象 ...

  2. 找出一堆数中最小的前K个数

    描写叙述: 给定一个整数数组.让你从该数组中找出最小的K个数 思路: 最简洁粗暴的方法就是将该数组进行排序,然后取最前面的K个数就可以. 可是,本题要求的仅仅是求出最小的k个数就可以,用排序能够但显然 ...

  3. PowerShell中和服务相关的命令

    New-Service https://docs.microsoft.com/en-us/powershell/module/microsoft.powershell.management/new-s ...

  4. 解决The requested resource is not available的办法

    1.问题描述: eclipse中使用tomcat来运行HelloWorld时出现The requested resource is not available. 在报错中有一行Setting prop ...

  5. 有关error PRJ0003错误的思考

    作者:朱金灿 来源:http://blog.csdn.net/clever101 今天同事遇到两个编译错误: 项目: error PRJ0003 : 生成"rc.exe"时出错. ...

  6. Linux常用的安全工具

    Linux常用的安全工具 "工欲善其事,必先利其器".作为一个合格的系统管理员,要应对可能发生的安全事件,掌握Linux下各种必须的安全工具是首要大事.本文主要介绍Linux上常用 ...

  7. [Chromium文档转载,第005章]Calling Mojo from Blink

    For Developers‎ > ‎Design Documents‎ > ‎Mojo‎ > ‎ Calling Mojo from Blink Variants Let's as ...

  8. Windows操作系统下将Redis安装为服务

    安装服务: E:/Redis/Redis-x64-3.2.100/redis-server.exe --service-install E:/Redis/Redis-x64-3.2.100/redis ...

  9. [Python] Python Libs

    The Python Standard Library has a lot of modules! To help you get familiar with what's available, he ...

  10. AutoLayout具体解释+手把手实战

    首先说一下这篇博客尽管是标记为原创,可是事实并不是本人亲自写出来的.知识点和样例本人花了一天各处查找和整理终于决定写一个汇总的具体解释,解去各位朋友到处盲目查找的必要,由于不是转载某一个人的内容.故此 ...