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 ...
随机推荐
- Vue slot插槽
插槽用于内容分发,存在于子组件之中. 插槽作用域 父级组件作用域为父级,子级组件作用域为子级,在哪定义的作用域就在哪. 子组件之间的内容是在父级作用域的,无法直接访问子组件里面的数据. 插槽元素 &l ...
- Nginx日志常用统计分析命令
IP相关统计 统计IP访问量(独立ip访问数量) awk '{print $1}' access.log | sort -n | uniq | wc -l 查看某一时间段的IP访问量(4-5点) gr ...
- asp.net mvc 5发布部署遇到403.14
asp.net mvc 5发布部署遇到403.14? HTTP错误 403.14 服务器配置为不列出此目录内容 除了设置.net运行的权限 isap和cgi启动状态外.可能是因为你手贱. 将这个钩去掉 ...
- wxpython 支持python语法高亮的自定义文本框控件的代码
在研发闲暇时间,把开发过程中比较重要的一些代码做个珍藏,下面的代码内容是关于wxpython 支持python语法高亮的自定义文本框控件的代码,应该是对大家也有用. import keywordimp ...
- Activi相关表归纳
Activi相关归纳总结记录: ACT_RE_* : 'RE'表示repository.这个前缀的表包含了流程定义和流程静态资源(图片,规则,等等). ACT_RU_* : 'RU'表示 ...
- Linux- 常用命令, Vim编辑器操作
1.Linux命令: ls >查看列表(蓝色为文件夹,白色为文件) ls -a >显示包括隐藏文件的所有文件 ls -l >以列表的形式显示 ls -lh >类似于ls -l ...
- SQL server 2017使用教程
1.安装: 从https://www.microsoft.com/en-us/sql-server/sql-server-downloads官网下载sql server2017试用版 180天 安装完 ...
- PYQT5登录界面跳转主界面方法
该问题,有很多种方法,但是很多方法要么这个有问题,要么那个有问题,最后终于找到一种没问题的方法.记录一下: 参考地址:https://www.jianshu.com/p/d18ff36a78d6?fr ...
- Delphi 拦截输入法输入结果
{ 拦截输入法输入的字符串.向编辑框中输入中文查看效果. Delphi XE7 } unit Unit1; interface uses Winapi.Windows, Winapi.Messages ...
- SQL MIN() 函数
MIN() 函数 MIN 函数返回一列中的最小值.NULL 值不包括在计算中. SQL MIN() 语法 SELECT MIN(column_name) FROM table_name 注释:MIN ...