题面: 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. Web.xml中四种验证方式

    源地址:https://blog.csdn.net/imimi_/article/details/78805642 <security-constraint> 的子元素 <http- ...

  2. Servlet常用的接口和类

    使用接口和类的作用:Servlet也是依靠继承父类和实现接口来实现的.使用Servlet必须要引入两个包:javax.servlet和javax.servlet.http.所有的Servlet应用都是 ...

  3. 常见的7种XSS

    1. URL Reflection 当URL以某种方式反映在源代码中时,我们可以添加自己的XSS向量/有效负载.对于PHP页面,可以使用斜杠字符(/)在页面名称之后添加任何内容 http://brut ...

  4. Linux安装配置Mariadb

    一.安装数据库Mariadb 参考:http://blog.51cto.com/12173069/2047746 从最新版本的linux系统开始,默认的是 Mariadb而不是mysql! 使用系统自 ...

  5. 章节十、5-CSS---用CSS 通配符定位元素

    以下演示操作以该网址中的输入框为例:https://learn.letskodeit.com/p/practice 一.css样式中有三种通配符“^.$.*” 语法:tag[attribute< ...

  6. Exchange 2010邮件服务器的搭建和部署

    Exchange主要是针对内部网或者企业网用户进行搭建的邮件服务器软件,利用它能够很快地搭建安全性较高的内部网邮件系统. 本次搭建在个人环境中实践,纯属爱好折腾,分四步骤,1.搭建windows 20 ...

  7. react-router(v4)

    概要 开发单页应用, 首先绕不开的内容就是路由, react router v4 版本是最新的版本. 和之前的版本相比, 成熟了很多, 也简单了很多, 使用起来更加方便. 核心 component r ...

  8. vue项目报错webpackJsonp is not defined

    在vue单页面应用中,我们大概都会使用CommonsChunkPlugin这个插件. 传送门 CommonsChunkPlugin 但是在项目经过本地测试没有任何问题,打包上线后却会报错 webpac ...

  9. Java 8 Stream介绍及使用2

    (原) stream中另一些比较常用的方法. 1. public static<T> Stream<T> generate(Supplier<T> s) 通过gen ...

  10. Win 7/10 安装Oracle 11g

    两个系统安装oracle的过程基本一致,注意安装时选桌面类(没有试过server,只有server类的操作系统选择server类) 安装过程:https://jingyan.baidu.com/alb ...