题面: 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 识别子串的更多相关文章

  1. BZOJ1396 识别子串【SAM+SegmentTree】

    BZOJ1396 识别子串 给定一个串\(s\),对于串中的每个位置,输出经过这个位置且只在\(s\)中出现一次的子串的最短长度 朴素的想法是,我们要找到那些只出现一次的子串,之后遍历每个串,把串所覆 ...

  2. bzoj千题计划318:bzoj1396: 识别子串(后缀自动机 + 线段树)

    https://www.lydsy.com/JudgeOnline/problem.php?id=1396 后缀自动机的parent树上,如果不是叶子节点,那么至少有两个子节点 而一个状态所代表子串的 ...

  3. BZOJ 1396: 识别子串( 后缀数组 + 线段树 )

    这道题各位大神好像都是用后缀自动机做的?.....蒟蒻就秀秀智商写一写后缀数组解法..... 求出Height数组后, 我们枚举每一位当做子串的开头. 如上图(x, y是height值), Heigh ...

  4. BZOJ.1396.识别子串(后缀自动机/后缀数组 线段树)

    题目链接 SAM:能成为识别子串的只有那些|right|=1的节点代表的串. 设这个节点对应原串的右端点为r[i],则如果|right[i]|=1,即\(s[\ [r_i-len_i+1,r_i-le ...

  5. BZOJ1396:识别子串(SAM)

    Description Input 一行,一个由小写字母组成的字符串S,长度不超过10^5 Output L行,每行一个整数,第i行的数据表示关于S的第i个元素的最短识别子串有多长. Sample I ...

  6. BZOJ-1396: 识别子串

    后缀自动机+线段树 先建出\(sam\),统计一遍每个点的\(right\)集合大小\(siz\),对于\(siz=1\)的点\(x\),他所代表的子串只会出现一次,设\(y=fa[x]\),则这个点 ...

  7. bzoj 1396 识别子串 后缀树+线段树

    题目大意 给定一个长度\(\le100000\)的字符串 求每一个位置的最短识别子串 对于位置\(x\),能识别子串\(s[i...j]\)的条件是 1.\(i\le x \le j\) 2.\(s[ ...

  8. BZOJ 1396:识别子串 SA+树状数组+单调队列

    1396: 识别子串 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 381  Solved: 243[Submit][Status][Discuss] ...

  9. BZOJ1396: 识别子串(后缀自动机,线段树)

    Description Input 一行,一个由小写字母组成的字符串S,长度不超过10^5 Output L行,每行一个整数,第i行的数据表示关于S的第i个元素的最短识别子串有多长. Sample I ...

随机推荐

  1. Vue slot插槽

    插槽用于内容分发,存在于子组件之中. 插槽作用域 父级组件作用域为父级,子级组件作用域为子级,在哪定义的作用域就在哪. 子组件之间的内容是在父级作用域的,无法直接访问子组件里面的数据. 插槽元素 &l ...

  2. Nginx日志常用统计分析命令

    IP相关统计 统计IP访问量(独立ip访问数量) awk '{print $1}' access.log | sort -n | uniq | wc -l 查看某一时间段的IP访问量(4-5点) gr ...

  3. asp.net mvc 5发布部署遇到403.14

    asp.net mvc 5发布部署遇到403.14? HTTP错误 403.14 服务器配置为不列出此目录内容 除了设置.net运行的权限 isap和cgi启动状态外.可能是因为你手贱. 将这个钩去掉 ...

  4. wxpython 支持python语法高亮的自定义文本框控件的代码

    在研发闲暇时间,把开发过程中比较重要的一些代码做个珍藏,下面的代码内容是关于wxpython 支持python语法高亮的自定义文本框控件的代码,应该是对大家也有用. import keywordimp ...

  5. Activi相关表归纳

    Activi相关归纳总结记录:        ACT_RE_* : 'RE'表示repository.这个前缀的表包含了流程定义和流程静态资源(图片,规则,等等). ACT_RU_* : 'RU'表示 ...

  6. Linux- 常用命令, Vim编辑器操作

    1.Linux命令: ls >查看列表(蓝色为文件夹,白色为文件) ls -a >显示包括隐藏文件的所有文件 ls -l >以列表的形式显示 ls -lh >类似于ls -l ...

  7. SQL server 2017使用教程

    1.安装: 从https://www.microsoft.com/en-us/sql-server/sql-server-downloads官网下载sql server2017试用版 180天 安装完 ...

  8. PYQT5登录界面跳转主界面方法

    该问题,有很多种方法,但是很多方法要么这个有问题,要么那个有问题,最后终于找到一种没问题的方法.记录一下: 参考地址:https://www.jianshu.com/p/d18ff36a78d6?fr ...

  9. Delphi 拦截输入法输入结果

    { 拦截输入法输入的字符串.向编辑框中输入中文查看效果. Delphi XE7 } unit Unit1; interface uses Winapi.Windows, Winapi.Messages ...

  10. SQL MIN() 函数

    MIN() 函数 MIN 函数返回一列中的最小值.NULL 值不包括在计算中. SQL MIN() 语法 SELECT MIN(column_name) FROM table_name 注释:MIN ...