#include <iostream>
#include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
#define maxn 100005
#define inf 100000000
using namespace std;
char st[maxn];
int n,tot,last,root,sum[maxn<<],tmp[maxn<<],fa[maxn<<],son[maxn<<][],dist[maxn<<],ri[maxn<<],pos[maxn<<];
struct Tsegment{
void prepare(){tot=last=root=,memset(ri,,sizeof(ri));}
int newnode(int x){
dist[++tot]=x; return tot;
}
void add(int op,int x){
int p=last,np=newnode(dist[p]+); last=np; ri[np]=; pos[np]=op;
for (;p&&!son[p][x];p=fa[p]) son[p][x]=np;
if (p==) fa[np]=root;
else{
int q=son[p][x];
if (dist[p]+==dist[q]) fa[np]=q;
else{
int nq=newnode(dist[p]+);
memcpy(son[nq],son[q],sizeof(son[q]));
fa[nq]=fa[q],fa[q]=fa[np]=nq;
for (;p&&son[p][x]==q;p=fa[p]) son[p][x]=nq;
}
}
}
}SAM;
struct Fsegment{
int l,r,lazy,val;
}tree1[maxn*];
struct date{
void build(int k,int l,int r){
tree1[k].lazy=tree1[k].val=inf; tree1[k].l=l,tree1[k].r=r;
if (l==r) return;
int mid=(l+r)/;
build(k*,l,mid),build(k*+,mid+,r);
}
void change(int k,int l,int r,int x,int y,int z){
if (x>y) return;
if (tree1[k].lazy!=inf){
if (tree1[k*].l) pushdown(k*,tree1[k].lazy);
if (tree1[k*+].l) pushdown(k*+,tree1[k].lazy);
tree1[k].lazy=inf;
}
if (l>=x&&r<=y){
pushdown(k,z);
return;
} int mid=(l+r)/;
if (x<=mid) change(k*,l,mid,x,y,z);
if (y>mid) change(k*+,mid+,r,x,y,z);
}
void pushdown(int k,int x){
tree1[k].lazy=min(tree1[k].lazy,x);
if (tree1[k].l==tree1[k].r) tree1[k].val=min(tree1[k].val,tree1[k].lazy);
}
int query(int k,int l,int r,int x){
if (tree1[k].lazy!=inf){
if (tree1[k*].l) pushdown(k*,tree1[k].lazy);
if (tree1[k*+].l) pushdown(k*+,tree1[k].lazy);
tree1[k].lazy=inf;
}
if (l==r&&r==x) return tree1[k].val;
int mid=(l+r)>>,ans=inf;
if (x<=mid) ans=min(ans,query(k*,l,mid,x));
else ans=min(ans,query(k*+,mid+,r,x));
return ans;
}
}Tree1;
struct Ksegment{
int l,r,val,lazy;
}tree[maxn*];
struct Graph{
void build(int k,int l,int r){
tree[k].l=l,tree[k].r=r,tree[k].lazy=tree[k].val=inf;
if (l==r) return; int mid=(l+r)/;
build(k*,l,mid),build(k*+,mid+,r);
}
void change(int k,int l,int r,int x,int y,int z){
if (x>y) return;
if (tree[k].lazy!=inf){
if (tree[k*].l) pushdown(k*,tree[k].lazy);
if (tree[k*+].l) pushdown(k*+,tree[k].lazy);
tree[k].lazy=inf;
}
if (l>=x&&r<=y){
pushdown(k,z);
return;
} int mid=(l+r)>>;
if (x<=mid) change(k*,l,mid,x,y,z);
if (y>mid) change(k*+,mid+,r,x,y,z);
}
void pushdown(int k,int x){
tree[k].lazy=min(tree[k].lazy,x);
if (tree[k].l==tree[k].r) tree[k].val=min(tree[k].val,tree[k].lazy);
}
int query(int k,int l,int r,int x){
if (tree[k].lazy!=inf){
if (tree[k*].l) pushdown(k*,tree[k].lazy);
if (tree[k*+].r) pushdown(k*+,tree[k].lazy);
tree[k].lazy=inf;
}
if (l==r&&r==x) return tree[k].val;
int mid=(l+r)>>,ans=inf;
if (x<=mid) ans=min(ans,query(k*,l,mid,x));
else ans=min(ans,query(k*+,mid+,r,x));
return ans;
}
}Tree;
int main(){
scanf("%s",st+),n=strlen(st+);
SAM.prepare();
for (int i=;i<=n;i++) SAM.add(i,st[i]-'a');
memset(sum,,sizeof(sum));
for (int i=;i<=tot;i++) sum[dist[i]]++;
for (int i=;i<=tot;i++) sum[i]+=sum[i-];
for (int i=;i<=tot;i++) tmp[sum[dist[i]]--]=i;
for (int i=tot,x;i>=;i--){
x=tmp[i];
if (fa[x]) ri[fa[x]]+=ri[x];
}
ri[root]=;
Tree1.build(,,n);
Tree.build(,,n);
for (int i=;i<=tot;i++){
if (ri[i]!=) continue;
int x=pos[i],y=dist[i],z=dist[fa[i]]+;
Tree1.change(,,n,x-y+,x-z+,x); //第一棵线段树按位置,记得减
Tree.change(,,n,x-z+,x,z); //第二棵线段树按长度
}
for (int i=;i<=n;i++){
printf("%d\n",min(Tree1.query(,,n,i)-i+,Tree.query(,,n,i)));
}
return ;
}

题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1396

题目大意:

做法;看到题目中所说的T在S中只出现过一次,就很容易想到用后缀自动机嘛,显然就是right值为一的状态,而且right值为一的状态只能是每次add时第一个新建的点,这很显然嘛,这就很方便记录了。然后再用线段树维护一下最小值,稍微想一下就行,当时我竟然是很快就想到了,不过我inf开小了,狂WA不止。

后缀自动机+线段树。

bzoj1396: 识别子串的更多相关文章

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

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

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

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

  3. BZOJ1396:识别子串(SAM)

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

  4. BZOJ-1396: 识别子串

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

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

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

  6. BZOJ1396 识别子串 和 BZOJ2865 字符串识别

    字符串识别 2865: 字符串识别 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 839  Solved: 261[Submit][Status][D ...

  7. BZOJ1396 识别子串 字符串 SAM 线段树

    原文链接http://www.cnblogs.com/zhouzhendong/p/9004467.html 题目传送门 - BZOJ1396 题意 给定一个字符串$s$,$|s|\leq 10^5$ ...

  8. BZOJ bzoj1396 识别子串

    题面: bzoj1396 题解: 先建出SAM,并计算right集合大小.显然符合条件的点的right集合大小为1. 对于每个right集合为1的状态显然可以算出这些状态的pos以及maxlen和mi ...

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

    题意 题目链接 Sol 后缀自动机+线段树 还是考虑通过每个前缀的后缀更新答案,首先出现次数只有一次,说明只有\(right\)集合大小为\(1\)的状态能对答案产生影响 设其结束位置为\(t\),代 ...

随机推荐

  1. Java调用C/C++编写的第三方dll动态链接库(zz)

    这里主要用的方法是JNI.在网上查资料时看到很多人说用JNI非常的复杂,不仅要看很多的文档,而且要非常熟悉C/C++编程.恐怕有很多人在看到诸如此类的评论时已经决定绕道用其他方法了.本文将做详细的介绍 ...

  2. java之yield(),sleep(),wait()区别详解

    1.sleep() 使当前线程(即调用该方法的线程)暂停执行一段时间,让其他线程有机会继续执行,但它并不释放对象锁.也就是说如果有synchronized同步快,其他线程仍然不能访问共享数据.注意该方 ...

  3. 实验二 Java面向对象程序设计

    实验二 Java面向对象程序设计 实验内容 1. 初步掌握单元测试和TDD 2. 理解并掌握面向对象三要素:封装.继承.多态 3. 初步掌握UML建模 4. 熟悉S.O.L.I.D原则 5. 了解设计 ...

  4. 命令行下 mysql 不是内部或外部命令排查方法

    首先确定你没有更改过MySQL的安装目录.如果你进行过改名或者更改了你的路径,那么要在相应的配置文件中更改你的你路径.找到C:\Windows\my.ini文件,更改你配置的文件路径,改成你修改后的路 ...

  5. 你误解 .net 了吗?

    我现在发现很多人对C#还存在很大的误解,例如C#是完全封闭的,C#不能跨平台,C#性能很差,C#不支持指针等等,持以上观点的人非常多,甚至最近看到的国内某机构对开发语言的统计中还写着C#不跨平台,不开 ...

  6. 新时代的coder如何成为专业程序员

    在移动互联网"泛滥"的今天,越来越多非专业(这里的非专业指的是非计算机专业毕业的程序员)程序员加入到了IT行业中来了,可能是因为移动互联网的火爆导致程序员容易就业而且工资很高,可能 ...

  7. vue.js+boostrap最佳实践

    一.为什么要写这篇文章 最近忙里偷闲学了一下vue.js,同时也复习了一下boostrap,发现这两种东西如果同时运用到一起,可以发挥很强大的作用,boostrap优雅的样式和丰富的组件使得页面开发变 ...

  8. ArcEngine将线符号化为立方体状

    对于二三维同步中的三维视图肯定是需要通过二维元素来符号化成三维元素的,之前项目测试临时采用这个自代的圆管状: esriSimple3DLineStyle AxisStyle = esriSimple3 ...

  9. js的单引号,双引号,转移符

    这里我们看到想在style后边在插入一个样式的变量,data.cssSytle.a是做边和邮编都是"",并且没有转移符

  10. dinic模板

    procedure addedge(u,v,cap:longint); begin sid[tot].u:=u; sid[tot].v:=v; sid[tot].cap:=cap; sid[tot]. ...