【JZOJ5093】【GDSOI2017第四轮模拟day3】字符串匹配 哈希
题面
对于一个字符集大小为C的字符串P,我们可以将任意两种字符在P中的位置进行互换,例如P=abcba,我们交换a,b就变为bacab,交换a,d就变为dbcbd,交换可以进行任意次。若交换后P变为了字符串Q,则我们称Q与P是匹配的。
现在给定两个字符集大小为C的字符串S,T,请你求出S中有多少个连续子串与T是匹配的。
100%的数据:1 <= n,m,C <= 10^6 , Case = 3
100
考虑哈希,
我们给一个长度为m的序列,规定它的哈希值为:\(\sum_{c \belong C}hh^{fi_c}*\sum_{T_i=c}HH^{i-fi_c}\)
那么维护这个哈希值就好了。
Code
#include<iostream>
#include<algorithm>
#include<math.h>
#include<stdio.h>
#include<string.h>
#define ll long long
#define fo(i,x,y) for(int i=x;i<=y;i++)
#define fd(i,x,y) for(int i=x;i>=y;i--)
using namespace std;
const int inf=0x7fffffff;
const char* fin="string.in";
const char* fout="string.out";
const int maxn=1000007;
const ll mo=1000000007ll;
int t,C,n,m,ans,a[maxn],fi[maxn],en[maxn],ne[maxn];
int dans[maxn];
ll h,H,hh[maxn],HH[maxn],nh[maxn],nH[maxn],_h[maxn],_H[maxn],_fi[maxn];
int read(){
int x=0;
char ch=getchar();
while (ch<'0' || ch>'9') ch=getchar();
while (ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
return x;
}
ll qpower(ll a,ll b){
ll c=1;
while (b){
if (b&1) c=c*a%mo;
a=a*a%mo;
b>>=1;
}
return c;
}
ll ni(ll v){return qpower(v,mo-2);}
void update(int de,int now){
if (_fi[de]<now){
int tmp=now-_fi[de];
_h[de]=_h[de]*nh[tmp]%mo;
_H[de]=_H[de]*nh[tmp]%mo;
_fi[de]=now;
}
}
int main(){
freopen(fin,"r",stdin);
freopen(fout,"w",stdout);
scanf("%d%d",&t,&C);
hh[0]=HH[0]=nh[0]=nH[0]=1;
ll hhh=ni(797ll),HHH=ni(648567ll);
fo(i,1,maxn-1){
hh[i]=797ll*hh[i-1]%mo,HH[i]=648567ll*HH[i-1]%mo;
nh[i]=hhh*nh[i-1]%mo,nH[i]=HHH*nH[i-1]%mo;
}
while (t--){
scanf("%d%d",&n,&m);
fo(i,1,n) a[i]=read();
h=H=0;
memset(fi,0,sizeof fi);
fo(i,1,m){
int j=read();
if (!fi[j]) H=(H+hh[fi[j]=i])%mo;
else H=(H+hh[fi[j]]*HH[i-fi[j]])%mo;
}
memset(fi,0,sizeof fi);
memset(ne,0,sizeof ne);
//memset(_h,0,sizeof _h);
fo(i,1,m){
int j=a[i];
if (!fi[j]){
h=(h+hh[fi[j]=en[j]=i])%mo;
_h[j]=_H[j]=hh[i];
_fi[j]=1;
}else{
ne[en[j]]=i;
en[j]=i;
h=(h+hh[fi[j]]*HH[i-fi[j]])%mo;
_h[j]=(_h[j]+hh[fi[j]]*HH[i-fi[j]])%mo;
}
}
dans[0]=0;
ans=(H==h);
if (H==h) dans[++dans[0]]=1;
fo(i,m+1,n){
int ad=a[i],de=a[i-m];
h=(h*nh[1])%mo;
//_h[de],_H[de]
update(de,i-m+1);
//
h-=_h[de];
if (ne[fi[de]]){
int tmp=ne[fi[de]]-fi[de];
fi[de]=ne[fi[de]];
_h[de]-=_H[de];
_H[de]=_H[de]*hh[tmp]%mo;
_h[de]=_h[de]*hh[tmp]%mo*nH[tmp]%mo;
}else fi[de]=0,_h[de]=0;
h+=_h[de];
//
if (!fi[ad]){
fi[ad]=en[ad]=i;
h=(h+hh[m])%mo;
_h[ad]=_H[ad]=hh[m];
_fi[ad]=i-m+1;
}else{
ne[en[ad]]=i;
en[ad]=i;
update(ad,i-m+1);
h=(h+_H[ad]*HH[i-fi[ad]])%mo;
_h[ad]=(_h[ad]+_H[ad]*HH[i-fi[ad]])%mo;
}
h=(h%mo+mo)%mo;
ans+=(H==h);
if (H==h) dans[++dans[0]]=i-m+1;
}
printf("%d\n",ans);
fo(i,1,dans[0]) printf("%d ",dans[i]);
printf("\n");
}
return 0;
}
【JZOJ5093】【GDSOI2017第四轮模拟day3】字符串匹配 哈希的更多相关文章
- 【JZOJ5094】【GDSOI2017第四轮模拟day3】鸽子 计算几何+floyd
题面 养鸽人要监视他的鸽子,有n只鸽子站在平面上,他可以在m个给定的点上设置监视器,如果一只鸽子在某个监视器上或者在两个监视器所连直线上或者在三个监视器所连直线的三角形内则其就咕咕咕了,现在养鸽人要让 ...
- CCF模拟题 字符串匹配
字符串匹配 时间限制: 1.0s 内存限制: 256.0MB 问题描述 给出一个字符串和多行文字,在这些文字中找到字符串出现的那些行.你的程序还需支持大小写敏感选项:当选项打开时,表示同一个字母的大写 ...
- 选拔赛 hash 字符串匹配 哈希算法(白书p374)
hash Description dr所在国度的有个奇怪的规定:他们的字母不是a~z,而是用1~1000表示. 利用这个奇怪的规定,dr想出了一个好玩的游戏:首先给出n个字符串(当然每个字符用1~ ...
- 字符串匹配的KMP算法
~~~摘录 来源:阮一峰~~~ 字符串匹配是计算机的基本任务之一. 举例来说,有一个字符串”BBC ABCDAB ABCDABCDABDE”,我想知道,里面是否包含另一个字符串”ABCDABD”? 许 ...
- {Reship}{KMP字符串匹配}
关于KMP字符串匹配的介绍和归纳,作者的思路非常清晰,推荐看一下 http://blog.csdn.net/v_july_v/article/details/7041827
- 字符串匹配(hash算法)
hash函数对大家来说不陌生吧 ? 而这次我们就用hash函数来实现字符串匹配. 首先我们会想一下二进制数. 对于任意一个二进制数,我们将它化为10进制的数的方法如下(以二进制数1101101为例): ...
- 【C++实现python字符串函数库】二:字符串匹配函数startswith与endswith
[C++实现python字符串函数库]字符串匹配函数startswith与endswith 这两个函数用于匹配字符串的开头或末尾,判断是否包含另一个字符串,它们返回bool值.startswith() ...
- sdut 2125串结构练习--字符串匹配【两种KMP算法】
串结构练习——字符串匹配 Time Limit: 1000ms Memory limit: 65536K 有疑问?点这里^_^ 题目链接:http://acm.sdut.edu.cn/sduto ...
- C语言字符串匹配函数
C语言字符串匹配函数,保存有需要时可以用: #include <stdio.h> #include <stdlib.h> #include <string.h> # ...
随机推荐
- 初识莫队——小Z的袜子
以前一直觉得莫队是多么高大上的一种算法,然而仔细看了下发现其实并不复杂,实质上就是技巧性的暴力美学. 在我看来莫队是一种分块排序后降低复杂度的算法,当答案可以通过左右端点一个一个移动维护出来的时候就可 ...
- Hbase实验:java创建和删除table
开启zookeeper.hadoop.hbase: 打开eclipse创一个java project,然后导入所需jar包: 写好java代码,运行create,然后去hbase shell里查看: ...
- Leetcode961. N-Repeated Element in Size 2N Array重复N次的元素
在大小为 2N 的数组 A 中有 N+1 个不同的元素,其中有一个元素重复了 N 次. 返回重复了 N 次的那个元素. 示例 1: 输入:[1,2,3,3] 输出:3 示例 2: 输入:[2,1,2, ...
- exit()和return语句的区别
(1)exit用于结束正在运行的程序,exit函数将参数是返回给OS.而return是返回函数值并退出函数. (2)return是语言级别的,它表示了调用堆栈的返回:而exit是系统调用级别的,它表示 ...
- 19-10-23-K-Aft
没改完题就过来沽博客是不是有点不好…… ZJ一下: 好好好题. T1数组大小…… $$10^7 \rightarrow 60$$ 事实上…… $$7 \times 10^7 \rightarrow 0 ...
- OSGi.NET使用笔记
一手资料来源于“开放工厂”,以下程序将会引用到一个核心文件UIShell.OSGi.dll 目前我对于OSGi这个框架的理解就是,主程序搜索并加载插件,以插件方式开放,便于扩展. 现在开始正式的旅程. ...
- 基于Skyline与ArcGIS Server的二三维联动功能实现
基于Skyline与ArcGIS Server的二三维联动功能实现主要利用WEB技术.ArcGIS for JavaScript.Skyline 二次开发以及ArcGIS 10.1 桌面工具. 利用A ...
- vue+element-ui 使用富文本编辑器
npm安装编辑器组件npm install vue-quill-editor –save 在components文件夹创建ue.vue组件,如下 ue.vue代码如下: <!-- 组件代码如下 ...
- Python学习之for循环--输出1-100中的偶数和登录身份认证
输出1-100中的偶数 效果图: 实现代码: for i in range(2,101,2): print(i,end = '\t') if(i == 34): print('\n') if (i = ...
- 关于html 制作table的一个注意点
数据分析,一般都需要显示数据,就需要使用html做复杂的表格.复杂表格一般是对td的rowspan .colspan属性值. 在html中<td> 标签定义 HTML 表格中的标准单元格. ...