2018.12.23 bzoj2865&&1396: 字符串识别(后缀自动机+线段树)
传送门
卡空间差评!
题意简述:给一个字串,对于每个位置求出经过这个位置且只在字串中出现一次的子串的长度的最小值。
解法:先建出samsamsam,显然只有当sizep=1size_p=1sizep=1的时候才对答案有贡献。
于是对于每个sizep=1size_p=1sizep=1的状态分情况更新答案。
- pos=[pos[p]−len[link[p]]+1,pos[p]]pos=[pos[p]-len[link[p]]+1,pos[p]]pos=[pos[p]−len[link[p]]+1,pos[p]],那么ansi=min{ansi,lenlinkp+1}ans_i=min\{ans_i,len_{link_p}+1\}ansi=min{ansi,lenlinkp+1}
- pos=[pos[p]−len[p]+1,pos[p]−len[link[p]]]pos=[pos[p]-len[p]+1,pos[p]-len[link[p]]]pos=[pos[p]−len[p]+1,pos[p]−len[link[p]]],那么ansi=min{ansi,lenp−i+1}ans_i=min\{ans_i,len_p-i+1\}ansi=min{ansi,lenp−i+1}
第一类直接上线段树。
第二类?我们令fi=ansi+if_i=ans_i+ifi=ansi+i,用线段树维护fif_ifi的值最后统计答案的时候减去iii即可。
代码:
#include<bits/stdc++.h>
#define ri register int
#define lc (p<<1)
#define rc (p<<1|1)
#define mid (l+r>>1)
using namespace std;
inline int read(){
int ans=0;
char ch=getchar();
while(!isdigit(ch))ch=getchar();
while(isdigit(ch))ans=(ans<<3)+(ans<<1)+(ch^48),ch=getchar();
return ans;
}
const int N=9e5+5;
int n,ans[500005];
char s[N];
struct SGT{
int mn[N<<1];
inline void build(int p,int l,int r,int f){
mn[p]=0x3f3f3f3f;
if(l==r){mn[p]=f?n+l:n;return;}
build(lc,l,mid,f),build(rc,mid+1,r,f);
}
inline void update(int p,int l,int r,int ql,int qr,int v){
if(ql>r||qr<l)return;
if(ql<=l&&r<=qr){mn[p]=min(mn[p],v);return;}
if(qr<=mid)update(lc,l,mid,ql,qr,v);
else if(ql>mid)update(rc,mid+1,r,ql,qr,v);
else update(lc,l,mid,ql,mid,v),update(rc,mid+1,r,mid+1,qr,v);
}
inline void query(int p,int l,int r,int f){
if(l==r){ans[l]=min(ans[l],mn[p]-f*l);return;}
mn[lc]=min(mn[lc],mn[p]),query(lc,l,mid,f);
mn[rc]=min(mn[rc],mn[p]),query(rc,mid+1,r,f);
}
}T1,T2;
struct SAM{
int last,tot,len[N],link[N],son[N][26],siz[N],rk[N],cnt[500005],pos[N];
SAM(){last=tot=1,len[0]=-1;}
inline void expand(int x,int id){
int p=last,np=++tot;
siz[last=np]=1,pos[np]=id,len[np]=len[p]+1;
while(p&&!son[p][x])son[p][x]=np,p=link[p];
if(!p){link[np]=1;return;}
int q=son[p][x],nq;
if(len[q]==len[p]+1){link[np]=q;return;}
len[nq=++tot]=len[p]+1,memcpy(son[nq],son[q],sizeof(son[q])),link[nq]=link[q],link[q]=link[np]=nq;
while(p&&son[p][x]==q)son[p][x]=nq,p=link[p];
}
inline void solve(){
for(ri i=1;i<=tot;++i)++cnt[len[i]];
for(ri i=1;i<=n;++i)cnt[i]+=cnt[i-1];
for(ri i=1;i<=tot;++i)rk[cnt[len[i]]--]=i;
T1.build(1,1,n,0),T2.build(1,1,n,1);
for(ri i=tot,p;i;--i){
p=rk[i];
if(siz[p]==1){
T1.update(1,1,n,pos[p]-len[link[p]]+1,pos[p],len[link[p]]+1);
T2.update(1,1,n,pos[p]-len[p]+1,pos[p]-len[link[p]],len[p]+1);
}
siz[link[p]]+=siz[p],pos[link[p]]=max(pos[link[p]],pos[p]);
}
fill(ans+1,ans+n+1,n+1);
T1.query(1,1,n,0),T2.query(1,1,n,1);
for(ri i=1;i<=n;++i)cout<<ans[i]<<'\n';
}
}sam;
int main(){
scanf("%s",s+1),n=strlen(s+1);
for(ri i=1;i<=n;++i)sam.expand(s[i]-'a',i);
sam.solve();
return 0;
}
2018.12.23 bzoj2865&&1396: 字符串识别(后缀自动机+线段树)的更多相关文章
- 【BZOJ4556】[TJOI2016&HEOI2016] 字符串(后缀自动机+线段树合并+二分)
点此看题面 大致题意: 给你一个字符串\(s\),每次问你一个子串\(s[a..b]\)的所有子串和\(s[c..d]\)的最长公共前缀. 二分 首先我们可以发现一个简单性质,即要求最长公共前缀,则我 ...
- BZOJ 2865 字符串识别 | 后缀数组 线段树
集训讲字符串的时候我唯一想出正解的题-- 链接 BZOJ 2865 题面 给出一个长度为n (n <= 5e5) 的字符串,对于每一位,求包含该位的.最短的.在原串中只出现过一次的子串. 题解 ...
- 【BZOJ-1396&2865】识别子串&字符串识别 后缀自动机/后缀树组 + 线段树
1396: 识别子串 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 312 Solved: 193[Submit][Status][Discuss] ...
- 模板—字符串—后缀自动机(后缀自动机+线段树合并求right集合)
模板—字符串—后缀自动机(后缀自动机+线段树合并求right集合) Code: #include <bits/stdc++.h> using namespace std; #define ...
- BZOJ1396: 识别子串(后缀自动机 线段树)
题意 题目链接 Sol 后缀自动机+线段树 还是考虑通过每个前缀的后缀更新答案,首先出现次数只有一次,说明只有\(right\)集合大小为\(1\)的状态能对答案产生影响 设其结束位置为\(t\),代 ...
- 洛谷P2178 [NOI2015]品酒大会(后缀自动机 线段树)
题意 题目链接 Sol 说一个后缀自动机+线段树的无脑做法 首先建出SAM,然后对parent树进行dp,维护最大次大值,最小次小值 显然一个串能更新答案的区间是\([len_{fa_{x}} + 1 ...
- 洛谷P4493 [HAOI2018]字串覆盖(后缀自动机+线段树+倍增)
题面 传送门 题解 字符串就硬是要和数据结构结合在一起么--\(loj\)上\(rk1\)好像码了\(10k\)的样子-- 我们设\(L=r-l+1\) 首先可以发现对于\(T\)串一定是从左到右,能 ...
- bzoj5417/luoguP4770 [NOI2018]你的名字(后缀自动机+线段树合并)
bzoj5417/luoguP4770 [NOI2018]你的名字(后缀自动机+线段树合并) bzoj Luogu 给出一个字符串 $ S $ 及 $ q $ 次询问,每次询问一个字符串 $ T $ ...
- BZOJ3413: 匹配(后缀自动机 线段树合并)
题意 题目链接 Sol 神仙题Orz 后缀自动机 + 线段树合并... 首先可以转化一下模型(想不到qwq):问题可以转化为统计\(B\)中每个前缀在\(A\)中出现的次数.(画一画就出来了) 然后直 ...
随机推荐
- @ResponseBody使用须知
-------------------siwuxie095 @ResponseBody 使用须知 使用 @ResponseBody 注解映射响应体 @ResponseBody 注解可被应用于方法上,标 ...
- ACM-ICPC 2018 沈阳赛区网络预赛 J. Ka Chang(树状数组+分块)
Given a rooted tree ( the root is node 1 ) of N nodes. Initially, each node has zero point. Then, yo ...
- Mybatis常用标签使用
trim元素的主要功能是可以在自己包含的内容前加上某些前缀,也可以在其后加上某些后缀,与之对应的属性是prefix和suffix:可以把包含内容的首部某些内容覆盖,即忽略,也可以把尾部的某些内容覆盖, ...
- android的四种线程池
使用线程池的好处: 首先通过线程池中线程的重用,减少创建和销毁线程的性能开销.其次,能控制线程池中的并发数,否则会因为大量的线程争夺CPU资源造成阻塞.最后,线程池能够对线程进行管理,比如使用Sche ...
- ubuntu下sudo apt-get update Sources 404 Not Found 解决方法
刚安装了ubuntu之后的主要安装命令无非就是apt-get install了,然而很多都在这里就夭折了. 使用apt-get install ***需要先执行apt-get update 加载文件包 ...
- 10-okHttp的同步与异步
我的理解如下: 同步: 实时的在等待返回结果: 异步:可以不是同步执行的,放入到执行队列中. 所以建议:如果需要根绝请求的结构做些判断应当用 同步,异步可能由于时间先后出现问题. /*post异步请求 ...
- springboot 日志2
SpringBoot关于日志的官方文档 1.简述 SpringBoot官方文档关于日志的整体说明 本博客基于SpringBoot_1.3.6大家请先简单看下这篇英文的官方文档,文中有说 Sprin ...
- export命令
http://blog.csdn.net/wl_fln/article/details/7258294 http://man.linuxde.net/export export命令 功能说明:设置或显 ...
- html5的鼠标拖拽
鼠标拖拽 <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF- ...
- sqlserver数据库的备份与还原——完整备份与还原
sqlserver提供四种数据库备份方式 完整备份:备份整个数据库的所有内容包括书屋和日志 差异备份:只备份上次完整备份后更高的数据部分 事务日志备份:只备份事务日志里的内容 文件或文件组备份:只备份 ...