Codeforce 535 D. Tavas and Malekas 解析(字串匹配)

今天我們來看看CF535D
題目連結

題目
給你一個字串$p$和一些$index$代表字串$p$在哪些位置會和長度為$n$的字串$s$匹配,求有多少種$s$的可能性。

前言

我還是只會$hash$

想法

$p$沒有覆蓋到的$index$數量記做$cnt$,如果那些會重疊的$p$沒有矛盾的話,那答案就是$26^$,因為有$cnt$的格子都是完全自由的。
這題有三種作法,都是要判斷重疊的$p$有沒有矛盾。注意,我們每次判斷一個從$y[i]$開始的$p$字串有沒有和前面重疊時,只需要和從$y[i-1]$開始的$p$字串比較就好,畢竟如果$y[i-2]$開始的$p$字串和$y[i-1]$開始的$p$字串沒有矛盾的話,那麼$y[i-2]$開始的$p$字串和$y[i]$開始的$p$字串就不會有矛盾(如果$y[i-1]$開始的字串和$y[i]$開始的字串沒有矛盾)。
三種匹配字串的方式是:

  1. Rolling hash,直接比對子字串的hash值看看一不一樣。
  2. KMP,只需要維護失敗函數,由於失敗函數代表一個最長的前綴的長度使得在$j-1$結束的字串的後綴和前綴一樣,那麼我們只要看看$p$字串的重疊部分是否沒有矛盾就好。
  3. Z-Algorithm,和KMP如出一轍,只不過這次$Z$函數代表從$j$開始往後的字串和整個字串的最長前綴長度。一樣只要看看$p$字串的重疊部分是否沒有矛盾就好。

程式碼(Rolling hash):

const int _n=1e6+10;
int t,n,m;
const int _N = _n; // str len
const int _p1 = 31, _M = 1000000009;
int pnM[_N] = {0}; // p^n mod M
int hp[_N] = {0}; // hash prefix
char p[_N];
int lenp,y[_n]; inline void genpnM() {
int res = 1; //p^0%M
pnM[0] = 1;
for (int i = 1; i < _N; i++) pnM[i] = res = (res * 1ll * _p1) % _M;
}
inline void genhp() { //hp[n]=hash prefix[0,n)
int res = 0;
hp[0] = 0; //[0,0) is empty string
for (int i = 1; i <= lenp; i++) hp[i] = res = (res * 1ll * _p1 + p[i - 1] * 1ll) % _M;
}
inline int dhash(char s[]) { //direct hash
int len = strlen(s);
int res = 0;
for (int i = 1; i <= len; i++) res = (res * 1ll * _p1 + s[i - 1] * 1ll) % _M;
return res;
}
inline int hashlr(int l, int r) { //[l,r)
int tmp = hp[r] * 1ll - pnM[r - l] * 1ll * hp[l] % _M;
if (tmp < 0) tmp += _M;
return tmp;
}
//以上是rolling hash模板
main(void) {ios_base::sync_with_stdio(0);cin.tie(0);cout.tie(0);
cin>>n>>m>>p;lenp=strlen(p);rep(i,0,m){cin>>y[i];y[i]--;}
genpnM();genhp();
int cnt=y[0];rep(i,1,m){
if(y[i-1]+lenp-1>=y[i]){
if(hashlr(y[i]-y[i-1],lenp)!=hashlr(0,lenp-(y[i]-y[i-1]))){
cout<<0<<'\n';return 0;
}
}
else cnt+=y[i]-(y[i-1]+lenp-1)-1;
}cnt+=n-(y[m-1]+lenp);
if(m==0)cnt=n;
cout<<powmod(26,cnt)<<'\n';
return 0;
}

標頭、模板請點Submission看
Submission

程式碼(KMP):

const int _n=1e6+10;
int t,n,m;
char p[_n];
int lenp,y[_n],jj,F[_n];
bool vis[_n];
main(void) {ios_base::sync_with_stdio(0);cin.tie(0);cout.tie(0);
cin>>n>>m>>p;lenp=strlen(p);rep(i,0,m){cin>>y[i];y[i]--;}
/*F[0]=-1,jj=-1;rep(i,0,lenp){
jj=F[i];
while(jj!=-1 and p[i]!=p[jj])jj=F[jj];
F[i+1]=jj+1; //if jj==-1, F[i]=0
}*/ //也行
F[0]=-1;
for(int i = 0, j = -1; i < lenp; F[++i] = ++j) { //計算失敗函數,一般來說i<lenp-1,但是這題我們需要i<lenp
while (j != -1 && p[i] != p[j]) j = F[j];
}
jj=F[lenp];
while(jj!=-1){ //看看哪些前綴可以和整條字串p的後綴匹配
vis[jj]=1;
jj=F[jj];
}
int cnt=y[0];rep(i,1,m){
if(y[i-1]+lenp-1>=y[i]){
if(!vis[lenp-(y[i]-y[i-1])]){
cout<<0<<'\n';return 0;
}
}
else cnt+=y[i]-(y[i-1]+lenp-1)-1;
}cnt+=n-(y[m-1]+lenp);
if(m==0)cnt=n;
cout<<powmod(26,cnt)<<'\n';
return 0;
}

標頭、模板請點Submission看
Submission

程式碼(Z-Algorithm):

const int _n=1e6+10;
int t,n,m,lenp,Z[_n],y[_n];
char p[_n];
main(void) {ios_base::sync_with_stdio(0);cin.tie(0);cout.tie(0);
cin>>n>>m>>p;lenp=strlen(p);rep(i,0,m){cin>>y[i];y[i]--;}
int L = 0, R = 0;
for (int i = 1; i < lenp; i++) {
if (i > R) {
L = R = i;
while (R < n && p[R-L] == p[R]) R++;
Z[i] = R-L; R--;
} else {
int k = i-L;
if (Z[k] < R-i+1) Z[i] = Z[k];
else {
L = i;
while (R < n && p[R-L] == p[R]) R++;
Z[i] = R-L; R--;
}
}
}
int cnt=y[0];rep(i,1,m){
if(y[i-1]+lenp-1>=y[i]){
if(Z[y[i]-y[i-1]]!=lenp-(y[i]-y[i-1])){
cout<<0<<'\n';return 0;
}
}
else cnt+=y[i]-(y[i-1]+lenp-1)-1;
}cnt+=n-(y[m-1]+lenp);
if(m==0)cnt=n;
cout<<powmod(26,cnt)<<'\n';
return 0;
}

標頭、模板請點Submission看
Submission

D. Tavas and Malekas 解析(字串匹配)的更多相关文章

  1. Codeforces 535D - Tavas and Malekas

    535D - Tavas and Malekas 题目大意:给你一个模板串,给你一个 s 串的长度,告诉你 s 串中有 m 个模板串并告诉你,他们的其实位置, 问你这样的 s 串总数的多少,答案对1e ...

  2. Codeforces Round #299 (Div. 2) D. Tavas and Malekas kmp

    题目链接: http://codeforces.com/problemset/problem/535/D D. Tavas and Malekas time limit per test2 secon ...

  3. JVM-class文件完全解析-字段表集合

     字段表集合 这个class文件的解析,分析得有点太久了.前面介绍类魔数,次版本号,主板本号,常量池入口,常量池,访问标志,类索引,父类索引和接口索引集合.下面就应该到字段表集合了.  紧接着接口索引 ...

  4. Tavas and Malekas

    题面 题目描述 给你两个字符串a和b,告诉所有你b在a中一定匹配的位置,求有中不同的字符串a.a的长度为n,b的长度为m,一定匹配的位置有p个.若b在a中的一定匹配的位置为x,说明a[x-x+m-1] ...

  5. codeforces 535D. Tavas and Malekas KMP

    题目链接 又复习了一遍kmp....之前都忘光了 #include<bits/stdc++.h> using namespace std; #define pb(x) push_back( ...

  6. CF #299 div1 B. Tavas and Malekas KMP-next数组

    题目链接:http://codeforces.com/contest/536/problem/B 一个原始字符串,一个未知字符串,每一次从pos[i]开始覆盖未知字符串,问最后字符串的形式,以及判断过 ...

  7. Codeforces Round #299 (Div. 2)D. Tavas and Malekas

    KMP,先预处理按每个节点标记,扫一遍更新每个匹配位置,最后kmp判断是否有重合而且不相同的地方 注意处理细节,很容易runtime error #include<map> #includ ...

  8. D. Tavas and Malekas DFS模拟 + kmp + hash || kmp + hash

    http://codeforces.com/contest/535/problem/D 如果真的要把m个串覆盖上一个串上面,是可以得,不会超时. 要注意到一点,全部覆盖后再判断时候合法,和边放边判断, ...

  9. 【Codeforces Round #299 (Div. 2) D】Tavas and Malekas

    [链接] 我是链接,点我呀:) [题意] 给你n个位置,然后让你从某些位置开始的|p|个位置,填上p这个字符串. 问你填的时候是否会发生冲突->输出0 否则输出最终n个位置组成的可能的字符串的总 ...

随机推荐

  1. kafka面试总结

    本文为复习期间面试总结 从以下方面对kafka面试进行总结:基本原理架构/项目实践/生产者/消费者/协调者/存储层/控制器 基本原理架构 简单讲下什么是kafka[一句话概括/架构图] 消息队列选型 ...

  2. ucore操作系统学习笔记(一) ucore lab1系统启动流程分析

    一.ucore操作系统介绍 操作系统作为一个基础系统软件,对下控制硬件(cpu.内存.磁盘网卡等外设),屏蔽了底层复杂多样的硬件差异:对上则提供封装良好的应用程序接口,简化应用程序开发者的使用难度.站 ...

  3. JMeter Websocket 二进制Binary压力测试或接口测试

    背景 最近在做游戏项目,做好java服务端,需要本地调试Websock服务,并且是二进制binary形式传输,网上的很多都是text形式传输的测试页面,所以不符合要求.为了解决websocket的二进 ...

  4. ksoap2-android的简单使用

    soap2-android 官网地址 https://simpligility.github.io/ksoap2-android/index.html 发行版本 https://oss.sonatyp ...

  5. Spring Cloud系列(二):Eureka应用详解

    一.注册中心 1.注册中心演变过程 2.注册中心必备功能 ① 服务的上线 ② 服务的下线 ③ 服务的剔除 ④ 服务的查询 ⑤ 注册中心HA ⑥ 注册中心节点数据同步 ⑦ 服务信息的存储,比如mysql ...

  6. 路由总结之静态、RIP、OSPF、IS-IS、BGP和策略路由

    路由无疑是当今网络的核心,看到浩如烟海的网络资料,可以让人皓首穷经啊,而且都是浩浩荡荡几百页,所以想搞简单点. 静态路由 静态路由无疑是最简单,也是最基本的. Ip route-static(指定是静 ...

  7. python opencv 读取图片 返回图片某像素点的b,g,r值

    转载:https://blog.csdn.net/weixin_41799483/article/details/80884682 #coding=utf-8   #读取图片 返回图片某像素点的b,g ...

  8. P4821 [中山市选]生成树

    题目链接 我们可以看一下题目中给的这张图. 首先,树是没有环的,所以我们要把所有的环上的边都删去一条. 我们可以现在每个五边形上删去一条边. 但删完之后我们会发现,里面还有一圈. 这时候,我们就要在这 ...

  9. RHSA-2018:3032-低危: binutils 安全和BUG修复更新

    [root@localhost ~]# cat /etc/redhat-release CentOS Linux release 7.2.1511 (Core) 修复命令: 使用root账号登陆She ...

  10. Docker笔记2:Docker 镜像仓库

    Docker 镜像的官方仓库位于国外服务器上,在国内下载时比较慢,但是可以使用国内镜像市场的加速器(比如阿里云加速器)以提高拉取速度. Docker 官方的镜像市场,可以和 Gitlab 或 GitH ...