这道题市面上就两种法:一种是SA+二分+主席树,一种是SAM+二分+主席树(有不少人打线段树合并???)(除此之外还有一种利用炒鸡水的数据的暴力SA,贼快.....)(当时学SA的时候没做这道题,现在早忘了SA了)
  分析题意:就是对于一个字符串,每次询问一个子串在另一个子串里能匹配上的最大前缀(非严格前缀)长度.
  我们知道,处理前缀的工具并不是十分充足,后缀倒是有一大帮,所以说把字符串倒过来,而把字符串倒过来是SAM处理问题时的常用技巧.现在,我们直接找答案,仍然很难找到一种时间复杂度合法的做法,那么我们发现这个东西,是具有可二分的性质的,所以我们二分.这时,我们的问题转化为了"给出某个子串的right和len,快速求出其在某一区间内有无相同子串",然后利用SAM的节点的性质,我们又可以把问题转化为"给出某个子串的right和len,询问其所在节点的right集合是否存在元素在某一区间",到此,我们的问题已经很清晰了.
  对于找到这个子串所在的节点,我们可以找到其right所对应的前缀所在的节点(这个我们在构造的时候就可以处理出来),然后在parent树上倍增找到包含有长度为len的子串的节点,接下来就是看在其right集合中是否存在元素在某个区间内,这个时候我们就可以搞出parent树的dfs序,并对dfs序建立主席树,主席树的线段树区间维护的是right集合.
  好了我们得到了一种O(nlog^2n)的优秀做法!!!

#include <cstdio>
#include <cstring>
#include <algorithm>
char xB[(<<)+],*xS=xB,*xT=xB;
#define gtc (xS==xT&&(xT=(xS=xB)+fread(xB,1,1<<15,stdin),xS==xT)?0:*xS++)
inline void read(int &x){
register char ch=gtc;bool symbol=false;
for(x=;ch<''||ch>'';ch=gtc)if(ch=='-')symbol=true;
for(;ch>=''&&ch<='';x=(x<<)+(x<<)+ch-'',ch=gtc);
if(symbol)x=-x;
}
const int N=;
const int A=;
const int B=;
char s[N],tmp[N];
int pos[N];
int trans[N<<][A],max[N<<],link[N<<];
int last,rt,sz;
#define newnode(a) (max[++sz]=(a),sz)
int id[N<<],dfn[N<<],L[N<<],R[N<<];
int n,m;
struct V{
int to,next;
}c[N<<];
int head[N<<],t;
inline void add(int x,int y){
c[++t].to=y,c[t].next=head[x],head[x]=t;
}
int f[N<<][B];
int Ti;
struct Segment_Tree{
Segment_Tree *ch[];
int size;
inline void* operator new (size_t);
}*root[N<<],*C,*mempool,*null;
#define mid ((l+r)>>1)
inline void* Segment_Tree:: operator new(size_t){
if(C==mempool){
C=new Segment_Tree[(<<)+];
mempool=C+(<<)+;
}
return C++;
}
inline void Init(){
null=new Segment_Tree;
null->ch[]=null->ch[]=null;
null->size=;
root[]=null;
last=rt=newnode();
}
int w,q,nw,nq;
inline void insert(int x){
w=last,nw=newnode(max[w]+);
while(w&&trans[w][x]==)trans[w][x]=nw,w=link[w];
if(!w)link[nw]=rt;
else if(max[w]+==max[trans[w][x]])
link[nw]=trans[w][x];
else{
q=trans[w][x],nq=newnode(max[w]+);
memcpy(trans[nq],trans[q],sizeof(trans[nq]));
while(w&&trans[w][x]==q)trans[w][x]=nq,w=link[w];
link[nq]=link[q],link[q]=link[nw]=nq;
}
last=nw,id[nw]=max[nw],pos[max[nw]]=nw;
}
inline void insert(Segment_Tree *&p,Segment_Tree *lst,int l,int r,int aim){
p=new Segment_Tree(),*p=*lst;
++p->size;if(l==r)return;
if(aim<=mid)insert(p->ch[],lst->ch[],l,mid,aim);
else insert(p->ch[],lst->ch[],mid+,r,aim);
}
inline int query(Segment_Tree *a,Segment_Tree *b,int l,int r,int z,int y){
if(z<=l&&r<=y)return b->size-a->size;
int ret=;
if(z<=mid)ret+=query(a->ch[],b->ch[],l,mid,z,y);
if(mid<y)ret+=query(a->ch[],b->ch[],mid+,r,z,y);
return ret;
}
inline void dfs(int x){
L[x]=++Ti,dfn[Ti]=x;
for(int i=;i<B;++i)f[x][i]=f[f[x][i-]][i-];
for(int i=head[x];i;i=c[i].next)
f[c[i].to][]=x,dfs(c[i].to);
R[x]=Ti;
}
inline int get(int x,int len){
for(int i=B-;i>=;--i)
if(max[f[x][i]]>=len)
x=f[x][i];
return x;
}
inline bool check(int st,int len,int l,int r){
if(r-l+<len)return false;
int x=get(pos[st],len);
return query(root[L[x]-],root[R[x]],,n,l+len-,r);
}
int main(){
//freopen("rio.in","r",stdin);
Init(),scanf("%d%d%s",&n,&m,tmp+);
register int i;
for(i=;i<=n;++i)s[i]=tmp[n-i+];
for(i=;i<=n;++i)insert(s[i]-'a');
for(i=;i<=sz;++i)add(link[i],i);dfs();
for(i=;i<=sz;++i){
root[i]=root[i-];
if(id[dfn[i]])insert(root[i],root[i-],,n,id[dfn[i]]);
}
int a,b,x,y,l,r,ans;
while(m--){
read(b),read(a),read(y),read(x);
a=n-a+,b=n-b+,x=n-x+,y=n-y+;
r=y-x+,l=,ans=;
while(l<=r)
if(check(y,mid,a,b))ans=mid,l=mid+;
else r=mid-;
printf("%d\n",ans);
}
return ;
}

【BZOJ 4556】[Tjoi2016&Heoi2016]字符串 SAM+二分+主席树的更多相关文章

  1. bzoj 4556 [Tjoi2016&Heoi2016]字符串——后缀数组+主席树

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4556 本来只要查 ht[ ] 数组上的前驱和后继就行,但有长度的限制.可以二分答案解决!然后 ...

  2. Bzoj 4556: [Tjoi2016&Heoi2016]字符串

    4556: [Tjoi2016&Heoi2016]字符串 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 177  Solved: 92[Sub ...

  3. [BZOJ4556][Tjoi2016&Heoi2016]字符串 后缀数组+主席树

    4556: [Tjoi2016&Heoi2016]字符串 Time Limit: 20 Sec  Memory Limit: 128 MB Description 佳媛姐姐过生日的时候,她的小 ...

  4. BZOJ 4556: [Tjoi2016&Heoi2016]字符串(后缀数组 + 二分答案 + 主席树 + ST表 or 后缀数组 + 暴力)

    题意 一个长为 \(n\) 的字符串 \(s\),和 \(m\) 个询问.每次询问有 \(4\) 个参数分别为 \(a,b,c,d\). 要你告诉它 \(s[a...b]\) 中的所有子串 和 \(s ...

  5. BZOJ 4556 [Tjoi2016&Heoi2016]字符串 ——后缀数组 ST表 主席树 二分答案

    Solution 1: 后缀数组暴力大法好 #include <map> #include <cmath> #include <queue> #include &l ...

  6. ●BZOJ 4556 [Tjoi2016&Heoi2016]字符串

    题链: http://www.lydsy.com/JudgeOnline/problem.php?id=4556 题解: 巨恶心...但是题很好呀,可以练习好几个比较麻烦的算法~ 1).预处理 首先用 ...

  7. bzoj 4552: [Tjoi2016&Heoi2016]排序【二分+线段树】

    二分值mid,然后把>=mid的赋值为1,其他赋值为0,每次排序就是算出区间内01的个数,然后分别把0和1放到连续的一段内,这些都可以用线段树来维护 二分的判断条件是操作完之后q位置上是否为1 ...

  8. 4556: [Tjoi2016&Heoi2016]字符串

    4556: [Tjoi2016&Heoi2016]字符串 链接 分析: 首先可以二分这个长度.此时需要判断是否存在一个以b结尾的前缀,满足与[c,d]的lcp大于等于mid. 如果我们把串翻转 ...

  9. [HEOI2016] 字符串 - 后缀数组,主席树,ST表,二分

    [HEOI2016] 字符串 Description 给定一个字符串 \(S\), 有 \(m\) 个询问,每个询问给定参数 \((a,b,c,d)\) ,求 \(s[a..b]\) 的子串与 \(s ...

随机推荐

  1. [转]win7下修改C盘USERS文件下的名称

    Win7下C:\Users\Cortana以账户名称命名的系统文件夹用户名的修改   Win7下C:\Users\Cortana以账户名称命名的系统文件夹用户名的修改 Win7下C:\Users\Co ...

  2. mysql更新返回值问题(更新内容跟之前内容一样,返回0)

    mysql更新返回值问题 问: 有一界面要更新个人信息,有几十个text标签需要填写假设有一用户从用户列表点修改进入了修改页面,但又没有修改什么,马上点击保存这时,因为text标签非常多,不能够一一判 ...

  3. YII2.0学习一 Advanced 模板安装

    下载github上的完事安装包(本机环境使用Composer安装非常慢) https://github.com/yiisoft/yii2-app-advanced 解压到文件目录 wwwroot/sh ...

  4. php-语言参考-类型3.2-未完待续

    一,PHP变量的8个类型 四种标量类型: boolean (布尔型) integer (整型) float (浮点型, 也称作 double) string (字符串) 两种复合类型: array ( ...

  5. 日志框架Log4j

    log4j是一个用Java编写的可靠,快速和灵活的日志框架(API),它在Apache软件许可下发布.Log4j已经被移植到了C,C++,C#,Perl,Python和Ruby等语言中. Log4j是 ...

  6. 04 mysql 基础三 (进阶)

    mysql 基础三 阶段一 mysql 单表查询 1.查询所有记录 select * from department; ​ select * from student; ​ select * from ...

  7. Python爬虫爬取百度翻译之数据提取方法json

    工具:Python 3.6.5.PyCharm开发工具.Windows 10 操作系统 说明:本例为实现输入中文翻译为英文的小程序,适合Python爬虫的初学者一起学习,感兴趣的可以做英文翻译为中文的 ...

  8. 005---json & pickle

    json & pickle 什么是序列化 序列化是指把内存里的数据类型转变成字符串,以便使其能存储在硬盘和网络传输.因为只能接收bytes类型. 为什么要序列化 持久化存储 分类 - json ...

  9. Eclipse_安装SAP_HANA数据库插件

    1.对于Eclipse Oxygen,请添加URL https://tools.hana.ondemand.com/oxygen 2.对于Eclipse luna,请添加URL     https:/ ...

  10. CSS3新特性回顾

    CSS3 介绍 开始实例 新特征简介 强大的CSS选择器 抛弃图片的视觉效果 盒模型变化(多列布局和弹性盒模型) 阴影效果 Web字体和web Font 图标 CSS33过渡与动画交互效果 媒体查询 ...