[CF700E][JZOJ5558]Cool Slogan (后缀自动机+线段树)
题意翻译
给出一个长度为$n$的字符串$s[1]$,由小写字母组成。定义一个字符串序列$s[1....k]$,满足性质:$s[i]$在$s[i-1]$ $(i>=2)$中出现至少两次(位置可重叠),问最大的$k$是多少,使得从$s[1]$开始到$s[k]$都满足这样一个性质。
题解
看了一个中午的代码终于弄懂了……$yyb$大佬强无敌……
一开始以为是SAM建好之后直接上$dp$,直接用$parent$树上的儿子节点更新父亲,因为parent树上节点不同说明出现次数必定不同。但交上去一发WA了。才发现自己这种想法不能保证$parent$树上的父亲必定在儿子中出现过超过两次……
还是来说说正解吧。我们先对原串建好SAM,并记录下每一个节点的任意一个$endpos$。考虑一下如何判断一个串是否在另一个串中出现,如果一个串(设串$A$)在另一个串(设串$B$)中出现了大于等于两次,那么在原串的任意位置的串$B$中都出现了两次。
于是考虑一下维护每一个点的$endpos$集合,这个只要用线段树就行了。如果$A$在$B$中出现了两次,那么$A$的$endpos$集合在$[pos[B]-len[B]+len[A],pos[B]]$中出现了至少两次(其中$pos[B]$表示$B$的任意一个$endpos$)。
不难发现有一个$dp$,每一个$parent$树上的父亲节点都可能转移到儿子节点,于是从上到下$dp$。又因为$parent$树上父亲是儿子的严格后缀,所以必然在儿子里出现了一次,那么只要考虑$endpos[A]$中是否有元素在$[pos[B]-len[B]+len[A],pos[B]-1]$中就行了
- //minamoto
- #include<cstdio>
- #include<cstring>
- using namespace std;
- template<class T>inline bool cmax(T&a,const T&b){return a<b?a=b,:;}
- const int N=;
- int fa[N],ch[N][],l[N],f[N],a[N],c[N],pos[N],top[N];char s[N];
- int last=,cnt=,n,ans=;
- void ins(int c,int k){
- int p=last,np=++cnt;last=np,l[np]=l[p]+,pos[np]=k;
- for(;p&&!ch[p][c];p=fa[p]) ch[p][c]=np;
- if(!p) fa[np]=;
- else{
- int q=ch[p][c];
- if(l[q]==l[p]+) fa[np]=q;
- else{
- int nq=++cnt;l[nq]=l[p]+,pos[nq]=pos[q];
- memcpy(ch[nq],ch[q],sizeof(ch[q]));
- fa[nq]=fa[q],fa[np]=fa[q]=nq;
- for(;ch[p][c]==q;p=fa[p]) ch[p][c]=nq;
- }
- }
- }
- void calc(){
- for(int i=;i<=cnt;++i) ++c[l[i]];
- for(int i=;i<=cnt;++i) c[i]+=c[i-];
- for(int i=;i<=cnt;++i) a[c[l[i]]--]=i;
- }
- int L[N*],R[N*],rt[N],tot;
- void modify(int &now,int l,int r,int x){
- now=++tot;if(l==r) return;
- int mid=l+r>>;
- if(x<=mid) modify(L[now],l,mid,x);
- else modify(R[now],mid+,r,x);
- }
- int merge(int x,int y){
- if(!x||!y) return x|y;
- int z=++tot;
- L[z]=merge(L[x],L[y]);
- R[z]=merge(R[x],R[y]);
- return z;
- }
- int query(int x,int l,int r,int ql,int qr){
- if(!x) return ;if(ql<=l&&qr>=r) return ;
- int mid=l+r>>;
- if(ql<=mid&&query(L[x],l,mid,ql,qr)) return ;
- if(qr>mid&&query(R[x],mid+,r,ql,qr)) return ;
- return ;
- }
- int main(){
- scanf("%d",&n);
- scanf("%s",s+);
- for(int i=;i<=n;++i) ins(s[i]-'a',i),modify(rt[last],,n,i);
- calc();
- for(int i=cnt;i>;--i) rt[fa[a[i]]]=merge(rt[fa[a[i]]],rt[a[i]]);
- for(int i=;i<=cnt;++i){
- int u=a[i],ff=fa[u];
- if(ff==){f[u]=,top[u]=u;continue;}
- int x=query(rt[top[ff]],,n,pos[u]-l[u]+l[top[ff]],pos[u]-);
- if(x) f[u]=f[ff]+,top[u]=u;
- else f[u]=f[ff],top[u]=top[ff];
- cmax(ans,f[u]);
- }
- printf("%d\n",ans);
- return ;
- }
[CF700E][JZOJ5558]Cool Slogan (后缀自动机+线段树)的更多相关文章
- 【CF700E】Cool Slogans 后缀自动机+线段树合并
[CF700E]Cool Slogans 题意:给你一个字符串S,求一个最长的字符串序列$s_1,s_2,...,s_k$,满足$\forall s_i$是S的子串,且$s_i$在$s_{i-1}$里 ...
- BZOJ3413: 匹配(后缀自动机 线段树合并)
题意 题目链接 Sol 神仙题Orz 后缀自动机 + 线段树合并... 首先可以转化一下模型(想不到qwq):问题可以转化为统计\(B\)中每个前缀在\(A\)中出现的次数.(画一画就出来了) 然后直 ...
- cf666E. Forensic Examination(广义后缀自动机 线段树合并)
题意 题目链接 Sol 神仙题Orz 后缀自动机 + 线段树合并 首先对所有的\(t_i\)建个广义后缀自动机,这样可以得到所有子串信息. 考虑把询问离线,然后把\(S\)拿到自动机上跑,同时维护一下 ...
- 洛谷P2178 [NOI2015]品酒大会(后缀自动机 线段树)
题意 题目链接 Sol 说一个后缀自动机+线段树的无脑做法 首先建出SAM,然后对parent树进行dp,维护最大次大值,最小次小值 显然一个串能更新答案的区间是\([len_{fa_{x}} + 1 ...
- BZOJ1396: 识别子串(后缀自动机 线段树)
题意 题目链接 Sol 后缀自动机+线段树 还是考虑通过每个前缀的后缀更新答案,首先出现次数只有一次,说明只有\(right\)集合大小为\(1\)的状态能对答案产生影响 设其结束位置为\(t\),代 ...
- [Luogu5161]WD与数列(后缀数组/后缀自动机+线段树合并)
https://blog.csdn.net/WAautomaton/article/details/85057257 解法一:后缀数组 显然将原数组差分后答案就是所有不相交不相邻重复子串个数+n*(n ...
- 洛谷P4493 [HAOI2018]字串覆盖(后缀自动机+线段树+倍增)
题面 传送门 题解 字符串就硬是要和数据结构结合在一起么--\(loj\)上\(rk1\)好像码了\(10k\)的样子-- 我们设\(L=r-l+1\) 首先可以发现对于\(T\)串一定是从左到右,能 ...
- 模板—字符串—后缀自动机(后缀自动机+线段树合并求right集合)
模板—字符串—后缀自动机(后缀自动机+线段树合并求right集合) Code: #include <bits/stdc++.h> using namespace std; #define ...
- 【BZOJ4556】[TJOI2016&HEOI2016] 字符串(后缀自动机+线段树合并+二分)
点此看题面 大致题意: 给你一个字符串\(s\),每次问你一个子串\(s[a..b]\)的所有子串和\(s[c..d]\)的最长公共前缀. 二分 首先我们可以发现一个简单性质,即要求最长公共前缀,则我 ...
- bzoj5417/luoguP4770 [NOI2018]你的名字(后缀自动机+线段树合并)
bzoj5417/luoguP4770 [NOI2018]你的名字(后缀自动机+线段树合并) bzoj Luogu 给出一个字符串 $ S $ 及 $ q $ 次询问,每次询问一个字符串 $ T $ ...
随机推荐
- sax解析xml文件,封装到对象中
创建User.java类 public class User { private String id; private String name; private String age; private ...
- 用python给图片添加文字(水印)
题目来源于:Python 练习册,每天一个小程序 第0000题 代码如下: #-*- coding:utf-8 -*- import PIL from PIL import Image from PI ...
- java程序员应该熟悉的20个有用的库(转)
优秀且经验丰富的Java开发人员的一个特点是API的广泛知识,包括JDK和第三方库.我花了很多时间学习API,特别是在阅读Effective Java 3rd Edition之后,Joshua Blo ...
- 695. Max Area of Island最大岛屿面积
[抄题]: 求最多的联通的1的数量 Given a non-empty 2D array grid of 0's and 1's, an island is a group of 1's (repre ...
- distributed lock manager (DLM)(分布式管理锁)
A distributed lock manager (DLM) provides distributed software applications with a means to synchron ...
- 53-C++ CH08 01
http://lx.lanqiao.cn/problem.page?gpid=T407 算法训练 C++ CH08 01 时间限制:1.0s 内存限制:256.0MB 问题描述 已 ...
- OpenCV之设计模式
参考资料: http://hahack.com/codes/opencv-and-mvc/ http://blog.csdn.net/yzhang6_10/article/details/508084 ...
- Sqlserver时间函数用法(二)
--1. 当前系统日期.时间 select getdate() --2015-01-06 09:27:27.277 --2.时间操作 dateadd 在向指定日期加上一段时间的基础上,返回新的 dat ...
- dll函数生成规则
[转]http://blog.csdn.net/beanjoy/article/details/9136127 所谓名字修饰约定,就是指变量名.函数名等经过编译后重新输出名称的规则. 比如源代码中函数 ...
- javascript总结19:javascript 使用概述
1 JS作用 1.验证表单(以前的网速慢)`` 2.页面特效(PC端的网页效果) 3.移动端(移动web和app) 4.异步和服务器交互(AJAX) 5.服务端开发(nodejs) 2 浏览器的主要构 ...