BZOJ bzoj1396 识别子串
题面: bzoj1396
题解:
先建出SAM,并计算right集合大小。显然符合条件的点的right集合大小为1.
对于每个right集合为1的状态显然可以算出这些状态的pos以及maxlen和minlen(fa的len+1)。
然后对于在pos和pos-minlen+1区间内的字符显然必须选长为minlen的一段区间。因此我们搞一棵线段树维护这些minlen,即对于在pos和pos-minlen+1区间内的字符在线段树上和minlen取min
对于另外的在pos-maxlen+1到pos-minlen+1的区间内,这些字符的区间长度都为\(pos-pos[i]\)(第二个pos[i]是每个字符的位置,第一个是该状态的pos),然后发现所有的这种情况都是\(pos\)减一个值,那么我们为了让其min,就另开一棵线段树维护每次pos的最小值。
最后统计答案即可。
#include<bits/stdc++.h>
#define ls (now<<1)
#define rs (now<<1|1)
using namespace std;
namespace Tzh{
const int maxn=4e5+10;
const int inf=INT_MAX;
int tot=1,last=1,c[maxn],a[maxn];
string S;
struct Suffix_AutoMaton{
int cnt,link,pos,len,ch[26];
}sam[maxn<<1];
struct segment_tree{
int ans[maxn];
struct Tr{
int lt,rt,tag;
}tree[maxn<<2];
void build(int now,int lt,int rt){
if(lt>rt) return ; tree[now].tag=inf;
tree[now].lt=lt,tree[now].rt=rt;
if(lt==rt) return ;
int mid=lt+rt>>1;
build(ls,lt,mid),build(rs,mid+1,rt);
}
void change(int now,int lt,int rt,int w){
if(tree[now].rt<lt||tree[now].lt>rt) return;
if(tree[now].lt>=lt&&tree[now].rt<=rt)
tree[now].tag=min(tree[now].tag,w);
else change(ls,lt,rt,w),change(rs,lt,rt,w);
}
void dfs(int now){
if(tree[now].lt==tree[now].rt){
ans[tree[now].lt]=tree[now].tag; return;
}
tree[ls].tag=min(tree[ls].tag,tree[now].tag);
tree[rs].tag=min(tree[rs].tag,tree[now].tag);
dfs(ls),dfs(rs);
}
}seg1,seg2;
void build(int x){ int cur=++tot,p=last;
sam[cur].len=sam[last].len+1; sam[cur].cnt=1;
sam[cur].pos=sam[cur].len-1; last=cur;
for(;p&&!sam[p].ch[x];p=sam[p].link) sam[p].ch[x]=cur;
if(!p) sam[cur].link=1;
else{ int q=sam[p].ch[x];
if(sam[q].len==sam[p].len+1) sam[cur].link=q;
else{ int clone=++tot;
sam[clone]=sam[q]; sam[clone].cnt=0;
sam[clone].len=sam[p].len+1;
for(;p&&sam[p].ch[x]==q;p=sam[p].link)
sam[p].ch[x]=clone;
sam[q].link=sam[cur].link=clone;
}
}
}
void cal(){
for(int i=1;i<=tot;i++) c[sam[i].len]++;
for(int i=1;i<=tot;i++) c[i]+=c[i-1];
for(int i=1;i<=tot;i++) a[c[sam[i].len]--]=i;
for(int i=tot;i;i--){ int p=a[i];
sam[sam[p].link].cnt+=sam[p].cnt;
}
}
void work(){
cin>>S;
for(int i=0;i<S.size();i++) build(S[i]-'a');
cal();seg1.build(1,0,S.size()),seg2.build(1,0,S.size());
for(int i=2;i<=tot;i++) if(sam[i].cnt==1){ int minlen=sam[sam[i].link].len+1;
seg1.change(1,sam[i].pos-minlen+1,sam[i].pos,minlen);
seg2.change(1,sam[i].pos-sam[i].len+1,sam[i].pos-minlen+1,sam[i].pos);
}
seg1.dfs(1),seg2.dfs(1);
for(int i=0;i<S.size();i++)
printf("%d\n",min(seg1.ans[i],seg2.ans[i]-i+1));
return ;
}
}
int main(){
#ifndef ONLINE_JUDGE
freopen("1396.in","r",stdin);
freopen("1396.out","w",stdout);
#endif
ios::sync_with_stdio(false);
Tzh::work();
return 0;
}
BZOJ bzoj1396 识别子串的更多相关文章
- BZOJ1396 识别子串【SAM+SegmentTree】
BZOJ1396 识别子串 给定一个串\(s\),对于串中的每个位置,输出经过这个位置且只在\(s\)中出现一次的子串的最短长度 朴素的想法是,我们要找到那些只出现一次的子串,之后遍历每个串,把串所覆 ...
- bzoj千题计划318:bzoj1396: 识别子串(后缀自动机 + 线段树)
https://www.lydsy.com/JudgeOnline/problem.php?id=1396 后缀自动机的parent树上,如果不是叶子节点,那么至少有两个子节点 而一个状态所代表子串的 ...
- BZOJ 1396: 识别子串( 后缀数组 + 线段树 )
这道题各位大神好像都是用后缀自动机做的?.....蒟蒻就秀秀智商写一写后缀数组解法..... 求出Height数组后, 我们枚举每一位当做子串的开头. 如上图(x, y是height值), Heigh ...
- BZOJ.1396.识别子串(后缀自动机/后缀数组 线段树)
题目链接 SAM:能成为识别子串的只有那些|right|=1的节点代表的串. 设这个节点对应原串的右端点为r[i],则如果|right[i]|=1,即\(s[\ [r_i-len_i+1,r_i-le ...
- BZOJ1396:识别子串(SAM)
Description Input 一行,一个由小写字母组成的字符串S,长度不超过10^5 Output L行,每行一个整数,第i行的数据表示关于S的第i个元素的最短识别子串有多长. Sample I ...
- BZOJ-1396: 识别子串
后缀自动机+线段树 先建出\(sam\),统计一遍每个点的\(right\)集合大小\(siz\),对于\(siz=1\)的点\(x\),他所代表的子串只会出现一次,设\(y=fa[x]\),则这个点 ...
- bzoj 1396 识别子串 后缀树+线段树
题目大意 给定一个长度\(\le100000\)的字符串 求每一个位置的最短识别子串 对于位置\(x\),能识别子串\(s[i...j]\)的条件是 1.\(i\le x \le j\) 2.\(s[ ...
- BZOJ 1396:识别子串 SA+树状数组+单调队列
1396: 识别子串 Time Limit: 10 Sec Memory Limit: 162 MBSubmit: 381 Solved: 243[Submit][Status][Discuss] ...
- BZOJ1396: 识别子串(后缀自动机,线段树)
Description Input 一行,一个由小写字母组成的字符串S,长度不超过10^5 Output L行,每行一个整数,第i行的数据表示关于S的第i个元素的最短识别子串有多长. Sample I ...
随机推荐
- 005. [转] SSH端口转发
玩转SSH端口转发 SSH有三种端口转发模式,本地端口转发(Local Port Forwarding),远程端口转发(Remote Port Forwarding)以及动态端口转发(Dynamic ...
- JS:onmouseover 、onmouseout
鼠标移动到P标签上时,改变文本和边框样式 <style type="text/css"> .onmouseover { color: red; border: 1px ...
- 处理范例代码Webapi中的Mongodb的Bson中ObjectId反序列化异常
微软代码范例中的一个Bug 处理Mongodb的Bson中ObjectId反序列化异常 https://docs.microsoft.com/zh-cn/aspnet/core/tutorials/f ...
- 一个GIS开源工具集架构的总结
文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/ 1.背景 最近由团队HWG主导的GIS开源工具集基本告一段落,该项目虽然 ...
- Android常用的四种布局(或者说是五种)
一.FrameLayout(帧布局): 显示特点:所有的子控件默认显示在FrameLayout的左上角,会重叠在一起显示. 常用属性: layout_gravity(设置给子控件,调整控件在容器内的重 ...
- 一些android开发实用性网站记录
android开发一些有用的网站有很多,可以方便我们开发,记录一下哈. 1.Android源代码在线阅读:https://www.androidos.net.cn/sourcecode 2.在线Jso ...
- USB_ModeSwitch for Android 7
USB_ModeSwitch官网: USB_ModeSwitch - Handling Mode-Switching USB Devices on Linux USB_ModeSwitch for A ...
- Canvas 绘图学习笔记2
1 绘制文本 fillText(string,x,y,maxWidth) //填充试绘制文本 strokeText(string,x,y,maxWidth) 画线试绘制文本 设置字体样式: cont ...
- Java日期的一些基本处理
今天工作中用到一些日期的处理.这里做一点浅显的整理. 1.日期的加减: 日期加减一般用到Calendar这个类比较好.这样不用处理12月加一个月和28.30.31.加一天等问题 String last ...
- C#标识符与关键字
标识符是指在程序中用来表示实物的单词,是分配给类型(类.结构.枚举.接口或委托).成员.变量或命名空间的名称.有效标识符必须遵循以下原则: 标识符不能以数字开头也不能包含空格: 标识符可以包含大小写字 ...