我又来复习\(kmp\)了

其实这道题主要是一个矩阵乘法,但是\(kmp\)在其中也有着非常重要的作用

我们可以这样定义状态\(dp[i][j]\)表示文本串进行到了\(i\)位置,同时文本串在最后和模式串匹配了一共\(j\)位的方案数

于是答案就是\(\sum_{i=0}^{m-1}dp[n][i]\)

之后我们想一下转移

显然\(dp[i]\)需要从\(i-1\)转移过来

但是怎么转移呢

有一个非常直观也非常\(sb\)的想法就是直接\(dp[i][j]=dp[i-1][j-1]\)

毕竟再补上模式串的第\(j\)位就可以啦

但是转移不止这些

那四个画出来的部分是\(next[j]\)

如果我们在转移\(dp[i][j]\)的状态的时候并不让文本串的第\(i\)位和模式串的第\(j\)位相等,而是在这里填上模式串的\(next[j]\)位置上的数

那么很显然\(dp[i-1][nx[j]]+=dp[i-1][j-1]\)

所以我们用\(kmp\)预处理出来一个这样的数组\(a[i][j]\)表示匹配到在模式串上匹配\(j\)向\(i\)转移的时候可以填几个数字

所以现在就有

\[dp[i][j]=\sum_{k=0}^{m-1}dp[i-1][k]*a[i][k]
\]

显然这是一个矩阵乘法就可以优化的柿子

现在的问题就变成了\(a\)数组怎么求

首先求出\(next\)数组,之后枚举这一位填什么,之后往前跳\(nx\),直到匹配就好了

代码

#include<iostream>
#include<cstring>
#include<cstdio>
#define re register
#define maxn 21
int dp[1001][maxn];
char S[maxn];
int n,m,mod;
int nx[maxn];
int ans[maxn][maxn];
int a[maxn][maxn];
inline void did_a()
{
int mid[maxn][maxn];
for(re int i=0;i<m;i++)
for(re int j=0;j<m;j++)
mid[i][j]=a[i][j],a[i][j]=0;
for(re int i=0;i<m;i++)
for(re int j=0;j<m;j++)
for(re int k=0;k<m;k++)
{
a[i][j]+=mid[i][k]*mid[k][j];
if(a[i][j]>=mod) a[i][j]%=mod;
}
}
inline void did_ans()
{
int mid[maxn][maxn];
for(re int i=0;i<m;i++)
for(re int j=0;j<m;j++)
mid[i][j]=ans[i][j],ans[i][j]=0;
for(re int i=0;i<m;i++)
for(re int j=0;j<m;j++)
for(re int k=0;k<m;k++)
{
ans[i][j]+=mid[i][k]*a[k][j];
if(ans[i][j]>=mod) ans[i][j]%=mod;
}
}
inline void quick(int b)
{
while(b)
{
if(b&1) did_ans();
b>>=1;
did_a();
}
}
int main()
{
scanf("%d%d%d",&n,&m,&mod);
scanf("%s",S+1);
nx[0]=nx[1]=0;
for(re int i=2;i<=m;i++)
{
int p=nx[i-1];
while(p&&S[p+1]!=S[i]) p=nx[p];
if(S[p+1]==S[i]) nx[i]=++p;
else nx[i]=0;
}
for(re int i=0;i<m;i++)
for(re char j='0';j<='9';j++)
{
int p=i;
while(p&&S[p+1]!=j) p=nx[p];
if(S[p+1]==j) a[p+1][i]++;
else a[0][i]++;
}
for(re int i=0;i<m;i++) ans[i][i]=1;
quick(n);
int tot=0;
for(re int i=0;i<m;i++)
tot=(tot+ans[i][0])%mod;
std::cout<<tot;
return 0;
}

【[HNOI2008]GT考试】的更多相关文章

  1. 1009: [HNOI2008]GT考试

    1009: [HNOI2008]GT考试 Time Limit: 1 Sec Memory Limit: 162 MB Description 阿申准备报名参加GT考试,准考证号为N位数\(X_1X_ ...

  2. 【bzoj1009】[HNOI2008]GT考试

    1009: [HNOI2008]GT考试 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 3018  Solved: 1856[Submit][Statu ...

  3. BZOJ_1009_[HNOI2008]_GT考试_(动态规划+kmp+矩阵乘法优化+快速幂)

    描述 http://www.lydsy.com/JudgeOnline/problem.php?id=1009 字符串全部由0~9组成,给出一个串s,求一个长度为n的串,不包含s的种类有多少. 分析 ...

  4. BZOJ 1009: [HNOI2008]GT考试( dp + 矩阵快速幂 + kmp )

    写了一个早上...就因为把长度为m的也算进去了... dp(i, j)表示准考证号前i个字符匹配了不吉利数字前j个的方案数. kmp预处理, 然后对于j进行枚举, 对数字0~9也枚举算出f(i, j) ...

  5. [HNOI2008] GT考试

    [HNOI2008] GT考试 标签 : DP 矩阵乘法 题目链接 题意 n位数中不出现一个子串的方案数. 题解 \(设dp[i][j]\)为前i位匹配到j时的合法方案数.(所谓合法,就是不能有别的匹 ...

  6. BZOJ_1009_[HNOI2008]GT考试_KMP+矩阵乘法

    BZOJ_1009_[HNOI2008]GT考试_KMP+矩阵乘法 Description 阿申准备报名参加GT考试,准考证号为N位数X1X2....Xn(0<=Xi<=9),他不希望准考 ...

  7. BZOJ1009 [HNOI2008]GT考试 矩阵

    去博客园看该题解 题目 [bzoj1009][HNOI2008]GT考试 Description 阿申准备报名参加GT考试,准考证号为N位数X1X2….Xn(0<=Xi<=9),他不希望准 ...

  8. BZOJ 1009 [HNOI2008]GT考试 (KMP + 矩阵快速幂)

    1009: [HNOI2008]GT考试 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 4266  Solved: 2616[Submit][Statu ...

  9. bzoj1009 / P3193 [HNOI2008]GT考试

    P3193 [HNOI2008]GT考试 设$f[i][j]$表示主串匹配到第$i$个位置,不吉利数字匹配到第$j$个位置 $g[i][j]$表示加上某数字使子串原来最多能匹配到第$i$个数字,现在只 ...

  10. bzoj 1009: [HNOI2008]GT考试 -- KMP+矩阵

    1009: [HNOI2008]GT考试 Time Limit: 1 Sec  Memory Limit: 162 MB Description 阿申准备报名参加GT考试,准考证号为N位数X1X2.. ...

随机推荐

  1. java 基础 --- 动态代理和静态代理

    问题  : 代理的应用场景是什么 动态代理的底层原理是什么,为什么只能继承接口 概述 代理模式是设计模式的一种,简单地说就是调用代理类的方法实际就是调用真实类的方法.这种模式在AOP (切面编程)中非 ...

  2. 使用TensorFlow进行中文自然语言处理的情感分析

    1 TensorFlow使用 分析流程: 1.1  使用gensim加载预训练中文分词embedding 加载预训练词向量模型:https://github.com/Embedding/Chinese ...

  3. 笔记本(华硕UL80VT)软件超频setFSB

    Warning !!!If you are a beginner, do not use this software. This software is for power users only. U ...

  4. 【C++并发实战】(三) std::future和std::promise

    std::future和std::promise std::future std::future期待一个返回,从一个异步调用的角度来说,future更像是执行函数的返回值,C++标准库使用std::f ...

  5. Struts 类型转换之局部和全局配置

    我们碰到过很多情况,就是时间日期常常会出现错误,这是我们最头疼的事,在struts2中有一些内置转换器,也有一些需要我们自己配置. 我们为什么需要类型转换呢? 在基于HTTP协议的Web应用中 客户端 ...

  6. 有标号的DAG计数系列问题

    传送门 II 设 \(f_i\) 表示 \(i\) 个点的答案 那么枚举至少 \(j\) 个点的出度为 \(0\) \[\sum_{j=0}^{i}(-1)^j\binom{i}{j}f_{i-j}2 ...

  7. Spring与Web整合

    一 概述 1.整合目的 将所有对象的创建与管理任务交给Spring容器,降低程序的耦合度. 2.整合途径 将Spring容器注入到Web容器中. 3.具体实现 使用ServletContextList ...

  8. 02.CSS选择器-->:focus

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  9. LintCode2016年8月22日算法比赛----平面列表

    平面列表 题目描述 给定一个列表,该列表中的每个要素要么是个列表,要么是整数.将其变成一个只包含整数的简单列表. 注意事项 如果给定的列表中的要素本身也是一个列表,那么它也可以包含列表. 样例 给定 ...

  10. java (Eclipse)连接MySQL数据库

    package com.ifruit.utils; import java.sql.Connection; import java.sql.DriverManager; import java.sql ...