@[DP, KMP, 矩陣快速冪]

Description

阿申准备报名参加GT考试,准考证号为\(N\)位数\(X_1 X_2 .. X_n(0 <= X_i <= 9)\),他不希望准考证号上出现不吉利的数字。

他的不吉利数学\(A_1 A_2 .. A_m (0 <= A_i <= 9)\)有M位,不出现是指\(X_1 X_2 .. X_n\)中没有恰好一段等于\(A_1 A_2 .. A_m\). \(A_1\)和\(X_1\)可以为\(0\)

Input

第一行输入\(N,M,K\).接下来一行输入\(M\)位的数。 \(N<=10^9,M<=20,K<=1000\)

Output

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

Sample Input

4 3 100

111

Sample Output

81

Solution

很容易想到DP方程

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

\[ans = \sum_{j = 0}^{m - 1}f[n][j]
\]

其中, \(f[i][j]\)表示考號第\(i\)位匹配到不吉利串第\(j\)位時的情況數; \(trans[k][j]\)記錄上一位位匹配至不吉利串中的第\(k\)位時, 填入\(num \in [1, 10)\)使得當前位匹配至不吉利串第\(j\)位的\(num\)數(實際上這個數量只能是\(1\)或者\(9\))

然後就會發現, \(i\)最大可以達到\(10^{9}\), 因此時間複雜度必須要優化.

想到矩陣快速冪, 發現可以直接將\(f\)整個省略掉, 只要求出\(trans^{n}\)即可

至於\(trans\)數組, 通過KMP算法與處理一下就好了

然後就可以直接看代碼了

#include<cstdio>
#include<cstring>
using namespace std;
const int M = 1 << 5;
int n, m, K;
char a[M];
int pre[M];
int trans[M][M];
int ans[M][M];
void mul(int a[M][M], int b[M][M], int res[M][M])
{
int tmp[M][M];
for(int i = 0; i < m; i ++)
for(int j = 0; j < m; j ++)
{
tmp[i][j] = 0;
for(int k = 0; k < m; k ++)
tmp[i][j] = (tmp[i][j] + a[i][k] * b[k][j]) % K;
}
for(int i = 0; i < m; i ++)
for(int j = 0; j < m; j ++)
res[i][j] = tmp[i][j];
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("BZOJ1009.in", "r", stdin);
freopen("BZOJ1009.out", "w", stdout);
#endif
scanf("%d%d%d", &n, &m, &K);
scanf("%s", a + 1);
for(int i = 1; i <= m; i ++)
*(a + i) -= '0';
pre[1] = 0;
for(int i = 2; i <= m; i ++)
{
int p = pre[i - 1];
while(p && (a[p + 1] != a[i]))
p = pre[p];
pre[i] = ((a[p + 1] == a[i]) ? (p + 1) : p);
}
memset(trans, 0, sizeof(trans));
for(int i = 0; i < m; i ++)
for(int j = 0; j < 10; j ++)
{
int p = i;
while(p && (a[p + 1] != j))
p = pre[p];
if(a[p + 1] == j)
p ++;
trans[p][i] = (trans[p][i] + 1) % K;
}
memset(ans, 0, sizeof(ans));
for(int i = 0; i < m; i ++)
ans[i][i] = 1;
while(n)
{
if(n & 1)
mul(ans, trans, ans);
mul(trans, trans, trans);
n >>= 1;
}
int sum = 0;
for(int i = 0; i < m; i ++)
sum = (sum + ans[i][0]) % K;
printf("%d", sum);
}

BZOJ1009GT考试 DP + KMP + 矩陣快速冪的更多相关文章

  1. B. Once Again... 解析(思維、DP、LIS、矩陣冪)

    Codeforce 582 B. Once Again... 解析(思維.DP.LIS.矩陣冪) 今天我們來看看CF582B 題目連結 題目 給你一個長度為\(n\)的數列\(a\),求\(a\)循環 ...

  2. 【bzoj1009】[HNOI2008]GT考试(矩阵快速幂优化dp+kmp)

    题目传送门:https://www.lydsy.com/JudgeOnline/problem.php?id=1009 这道题一看数据范围:$ n<=10^9 $,显然不是数学题就是矩乘快速幂优 ...

  3. 【BZOJ1009】GT考试(KMP算法,矩阵快速幂,动态规划)

    [BZOJ1009]GT考试(KMP算法,矩阵快速幂,动态规划) 题面 BZOJ 题解 看到这个题目 化简一下题意 长度为\(n\)的,由\(0-9\)组成的字符串中 不含串\(s\)的串的数量有几个 ...

  4. 1235: 入学考试[DP]

    1235: 入学考试 [DP] 时间限制: 1 Sec 内存限制: 128 MB 提交: 37 解决: 12 统计 题目描述 辰辰是个天资聪颖的孩子,他的梦想是成为世界上最伟大的医师.为此,他想拜附近 ...

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

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

  6. [Bzoj1009][HNOI2008]GT考试(KMP)(矩乘优化DP)

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

  7. [HNOI2008] GT考试(DP+矩阵快速幂+KMP)

    题目链接:https://www.luogu.org/problemnew/show/P3193#sub 题目描述 阿申准备报名参加 GT 考试,准考证号为 N 位数 X1,X2…Xn(0 <= ...

  8. 2018.10.22 bzoj1009: [HNOI2008]GT考试(kmp+矩阵快速幂优化dp)

    传送门 f[i][j]f[i][j]f[i][j]表示从状态"匹配了前i位"转移到"匹配了前j位"的方案数. 这个东西单次是可以通过跳kmp的fail数组得到的 ...

  9. [BZOJ1009][HNOI2008]GT考试 DP+矩阵快速幂+KMP

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1009 我们令$dp(i,j)$表示已经填了$i$位,而且后缀与不幸运数字匹配了$j$位,那 ...

随机推荐

  1. IDEA入门学习笔记1:资料收集

    IDEA2018软件下载 :https://mp.weixin.qq.com/s?__biz=MzIwMjE1MjMyMw==&mid=2650200056&idx=1&sn= ...

  2. STM32F407VET6之IAR之ewarm7.80.4工程建立(基于官方固件库1.6版本)

    今天把stm32F407的工程之IAR建立完成了,特此记录下. 下载官方固件库,STM32F4xx_DSP_StdPeriph_Lib_V1.6.1,V1.8.0版本的同理.新建以下几个文件 src放 ...

  3. 线段树:POJ3468-A Simple Problem with Integers(线段树注意事项)

    A Simple Problem with Integers Time Limit: 10000MS Memory Limit: 65536K Description You have N integ ...

  4. BZOJ 2687: 交与并

    答案存在于 1.两个互相包含的区间 2.两个互不包含的区间 决策单调性显然 但是这代码很精妙啊,并不知道这个为什么能这样写 #include<cstdio> #include<alg ...

  5. Python之code对象与pyc文件(一)

    Python程序的执行过程 我们都知道,C语言在执行之前需要将源代码编译成可执行的二进制文件,也就是将源代码翻译成机器代码,这种二进制文件一旦生成,即可用于执行.但是,Python是否一样呢?或许很多 ...

  6. meta-data

    <meta-data android:name="string"   android:resource="resource specification"  ...

  7. 04_ThreadLocal整合事务操作

    文章导读: 本文主要讲解了如何在没有框架情况下如何解决Dao的事务问题, 重点理解Connection存放到WeakReference中为什么垃圾回收的时候Connection不回收 视频与源码下载: ...

  8. ptyhon - 接口自动化测试实战case1

    work_20181203_httprequest.py: import requestsclass http_request: def http_get(url,params): res = req ...

  9. 正则表达式 去除所有非ASCII字符

    需求: 去除字符串中包含的所有外国字符 只能使用正则如下,找到包含非ASCII的记录 db=# select * from test where info ~ '[^(\x00-\x7f)]'; id ...

  10. 微信小程序开发 -- 设置屏幕亮度

    wx.setScreenBrightness(OBJECT) 设置屏幕亮度. OBJECT参数说明: 参数 类型 必填 说明 value Number 是 屏幕亮度值,范围 0~1,0 最暗,1 最亮 ...