[JZOJ5897]密匙--哈希骚操作
[JZOJ5897]密匙--哈希骚操作
题目链接
太懒了自行Google
前置技能
二分/倍增求LCP
e.g TJOI2017DNA
分析
这题看了样例解释才知道什么意思
本以为自己身为mo法师蛤希已经掌握的差不多,做了这道题才发现我还是Too Young Too Simple
考虑字符集只有\({a,b}\)怎么做,我们每次二分同时判断要不要转化就好了
std就很强了,我们对每一个字母ch,都对\(S\)串跑遍01蛤希(瞎编的名字)
\(ha[i][ch-'a']=ha[i-1][ch-'a']*w+[s[i]==ch]\)
同时再对\(T\)串跑一遍正常的蛤希,\(ha_t[i]=ha_t[i-1]*w + t[i]\)
这时候我们还是进行二分/倍增求LCP,但怎么判断\(S\)串一个前缀能不能得到\(T\)串对应前缀呢
我们先用\(set\)预处理一下,也就是在对应字符那里插入它在\(S\)中出现的位置,这样我们就能较快的求出一个区间内某个字符出现的位置
考虑这时候我们在判断\(S\)从\(st\)开始的前缀中\([st,st+len]\)部分能否转化为\(T\)串对应部分
我们还是遍历每一个字母,找到在这个区间内出现的一个位置\(pos\)(如果没有就不管),再判断是否需要映射成\(T\)串对应位置的字符,如果需要而且它之前没有被占用(也就是之前没有出现在字母对中),那就对\(T[pos]\)和\(S[pos]\)建立映射关系
同时我们求出这个区间内\(S\)串关于字母\(S[pos]\)的哈希值(如果不知道怎么求的先去学一下\(Rabin-Karp Hash\)),乘上\(T[pos]\)
用一个数\(sum\)累加所有出现字母的这种乘积哈希值,如果没有出现映射矛盾的话,我们会发现如果可以转化成\(T\)串一个前缀,那么这个\(sum\)是会等于\(T\)串对应前缀的哈希值,这是因为\(S\)对应字母的哈希值只有01表示有没有出现,乘上字母的ASIIC值就是该字母在这个区间内的哈希值
代码
/*
code by RyeCatcher
*/
inline char gc(){
static char buf[SIZE],*p1=buf,*p2=buf;
return p1==p2&&(p2=(p1=buf)+fread(buf,1,SIZE,stdin),p1==p2)?EOF:*p1++;
}
#define gc getchar
template <class T>inline void read(T &x){
x=0;int ne=0;char c;
while((c=gc())>'9'||c<'0')ne=c=='-';x=c-48;
while((c=gc())>='0'&&c<='9')x=(x<<3)+(x<<1)+c-48;x=ne?-x:x;return ;
}
const int maxn=100005;
const int inf=0x7fffffff;
ull ha[maxn][28],hat[maxn],mi[maxn];
set <int> pos[28];
int n,m;
int vis[28];
char s[maxn],t[maxn];
bool chk(int l,int r){
ull tmp=0;
set<int>::iterator it;
int x,y;
memset(vis,-1,sizeof(vis));
for(ri i=0;i<26;i++){
it=pos[i].lower_bound(l);
if(it==pos[i].end()||*it>r)continue;
x=*it,y=t[x-l+1]-'a';
if(vis[i]!=-1||vis[y]!=-1){//判断映射是否矛盾
if(vis[y]!=i||vis[i]!=y){
return 0;
}
}
vis[i]=y,vis[y]=i;
tmp+=(ha[r][i]-ha[l-1][i]*mi[r-l+1])*t[x-l+1];//取出字母的区间哈希值
}
if(tmp==hat[r-l+1])return 1;
return 0;
}
int main(){
FO(key);
read(n),read(m);
scanf("%s",s+1);
scanf("%s",t+1);
for(ri k=0;k<26;k++){
ull tmp=0;
char o='a'+k;
for(ri i=1;i<=n;i++){
if(s[i]!=o)tmp=tmp*31;
else tmp=tmp*31+1;
ha[i][k]=tmp;
}
}
mi[0]=1;
for(ri i=1;i<=n;i++)pos[s[i]-'a'].insert(i),mi[i]=mi[i-1]*31;//预处理
for(ri i=1;i<=m;i++){
hat[i]=hat[i-1]*31+t[i];
}
for(ri o=1;o<=n;o++){
int k=0,p=1;
while(p){
if(o+k+p<=n&&chk(o,o+k+p))k+=p,p=p<<1;
else p=p>>1;
while(o+k+p>n||k+p>m)p=p>>1;
}
printf("%d\n",1+k);
}
return 0;
}
[JZOJ5897]密匙--哈希骚操作的更多相关文章
- ssh密匙互信操作【原创】
1.简便ssh密匙信任方法 只在一台服务器上创建ssh-keygen [root@SMSJKSRVBJ02 ~]# ssh-keygen Generating public/private rsa k ...
- 如何在命令长度受限的情况下成功get到webshell(函数参数受限突破、mysql的骚操作)
0x01 问题提出 还记得上篇文章记一次拿webshell踩过的坑(如何用PHP编写一个不包含数字和字母的后门),我们讲到了一些PHP的一些如何巧妙地绕过数字和字母受限的技巧,今天我要给大家分享的是如 ...
- 实现liunx之间无密码访问——ssh密匙
环境描述 两台linux服务器 172.16.1.22[client],172.16.1.33[server],想要实现client服务器ssh无密码访问server服务器. 使用技术 linux 的 ...
- Git科普文,Git基本原理&各种骚操作
Git简单介绍 Git是一个分布式版本控制软件,最初由Linus Torvalds创作,于2005年以GPL发布.最初目的是为更好地管理Linux内核开发而设计. Git工作流程以及各个区域 Work ...
- Ubu18远程登录密匙设置
Ubu18设置远程密匙登录 相关文件 /etc/ssh/sshd_config 注意vscode使用博客园插件需要进行端口转发,在vscode端口处设置41385 本地生成密匙,任选一种,这里只介绍第 ...
- 多节点ssh免密匙登录
1,在所有节点上,使用yourname用户名执行: ssh-keygen -t dsa -P '' -f /home/yourname/.ssh/id_dsa 2,在node1的/home/yourn ...
- SQLServer2008R2企业版密匙
SQLServer2008R2企业版密匙: R88PF-GMCFT-KM2KR-4R7GB-43K4B
- visual studio无法输入密匙解决方法
控制面板->程序和功能->修复/卸载(更改),当到输入密匙界面运行程序,即可出现密匙输入框. 所用程序网上搜索:CrackVS2008ForWindows7
- Linux学习(四)单用户模式、救援模式、虚拟机克隆、linux互连(包括密匙登录)
一.单用户模式 忘记root密码后,找回密码有两种方法: 单用户(grub没有加密的情况下可以使用) 救援模式 这一节我们先讲单用户模式 1.先重启(3种方法) reboot init 6 sho ...
随机推荐
- MySQL 8.0: From SQL Tables to JSON Documents (and back again)
MySQL 8.0: From SQL Tables to JSON Documents (and back again) | MySQL Server Bloghttps://mysqlserver ...
- MongoDB基础笔记
MongoDB show dbs 查看当前的数据库 use test 选库 show tables/collections 查看当前库下的文档 db.help() 查看帮助 db.createColl ...
- SQL-W3School-总结:SQL 快速参考
ylbtech-SQL-W3School-总结:SQL 快速参考 1.返回顶部 1. 来自 W3School 的 SQL 快速参考.可以打印它,以备日常使用. SQL 语句 语句 语法 AND / O ...
- ISO/IEC 9899:2011 条款6.2.4——对象的存储持久性
6.2.4 对象的存储持久性 1.一个对象具有一个存储持久性来确定其生命周期.一共有四种存储持久性:静态的,线程的,自动的,以及分配的.分配存储在7.22.3中描述. 2.一个对象的生命周期是程序执行 ...
- css简单学习属性3---css属性选择器
1:通配符 <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF ...
- 在Excel多个工作表间快速切换的绝招
在Excel多个工作表间快速切换的绝招 几乎每个Excel用户"数据分析师"都应该知道,如果一个Excel工作簿中包括许多个工作表,我们"数据分析师"可以通过单 ...
- 【ARM-Linux开发】【QT开发】Couldnt load module input, no modules loaded
export TSLIB_FBDEVICE=/dev/fb0 export TSLIB_PLUGINDIR=/usr/lib/ts (input.so 所在目录)
- 优化apk的odex处理
前文讲到了apk的反编译流程,但在有的情况下会出现apk的资源文件和代码文件分开的情况,这种情况会出现如下目录结构,并且在apk文件里面并没有dex文件. 所需工具 smali.jar和baksmal ...
- Evaluating Automatically Generated timelines from the Web (paper1)
摘要: 问题:There is a need that 以一个更全面/更综合的方式来展现搜索结果.对此,作者正在开发一个系统,called “Cronopath”,这个系统将产生一个时间线,通过决定每 ...
- docker管理工具lazydocker
docker管理工具lazydocker 简介 这是一个为了能再终端中更方便管理docker的工具 项目地址 https://github.com/jesseduffield/lazydocker 安 ...