题意:给定主串s和m个模式串,每次询问[l,r]的模式串中出现s[pl...pr]次数最多的串和次数。

这题挺简单的,先把所有模式串拿来建广义后缀自动机,询问相当于子树众数,用线段树合并即可。

那我为什么写这题题解呢?

1.作为我博客第一道非BZOJ题。

2.作为我博客第一道写题意(英文)的题。

3.这题是老师留的作业,所以可以骗访问量。

4.记录一下广义后缀自动机的优秀写法,以前我是无脑建的。

#include <cstdio>
#include <algorithm>
using namespace std;
#define l(x) t[x].l
#define r(x) t[x].r
#define M ((L+R)>>1) typedef pair<int,int> pr;
const int S=,N=;
char s[N],sr[S];
int m,e,q,l1,r1,l2,r2,ls,sz,tt,po[S],ln[S],l[N],f[N],ch[N][],hd[N],nx[N],to[N],rt[N],fa[N][];
struct nd {int l,r; pr p;}t[N*];
void ad(int x,int y) {to[++e]=y,nx[e]=hd[x],hd[x]=e;} void ins(int c) {
int u=ls,x=ch[u][c];
if(x) {
if(l[u]+==l[x]) {ls=x; return;}
l[++sz]=l[u]+,f[sz]=f[x],f[x]=sz,ls=sz;
for(int j=;j<;j++) ch[sz][j]=ch[x][j];
for(;u&&ch[u][c]==x;u=f[u]) ch[u][c]=sz;
if(ch[u][c]==x) ch[u][c]=sz;
} else {
for(ls=++sz,l[sz]=l[u]+;u&&!ch[u][c];u=f[u]) ch[u][c]=sz;
int x=ch[u][c];
if(!x) {ch[u][c]=sz; return;}
if(l[u]+==l[x]) {f[sz]=x; return;}
l[++sz]=l[u]+,f[sz]=f[x],f[x]=f[ls]=sz;
for(int j=;j<;j++) ch[sz][j]=ch[x][j];
for(;u&&ch[u][c]==x;u=f[u]) ch[u][c]=sz;
if(ch[u][c]==x) ch[u][c]=sz;
}
} void pu(int x) {t[x].p=max(t[l(x)].p,t[r(x)].p);}
int mrg(int x,int y,int L=,int R=m) {
if(!x||!y) return x|y;
int z=++tt;
if(L==R) {t[z].p=make_pair(t[x].p.first+t[y].p.first,-L); return z;}
l(z)=mrg(l(x),l(y),L,M),r(z)=mrg(r(x),r(y),M+,R),pu(z);
return z;
}
void upd(int &x,int v,int L=,int R=m) {
if(!x) x=++tt;
if(L==R) {t[x].p=make_pair(t[x].p.first+,-v); return;}
if(v<=M) upd(l(x),v,L,M); else upd(r(x),v,M+,R);
pu(x);
}
pr qr(int x,int l,int r,int L=,int R=m) {
if(!x) return make_pair(,);
if(l<=L&&r>=R) return t[x].p;
if(r<=M) return qr(l(x),l,r,L,M); if(l>M) return qr(r(x),l,r,M+,R);
return max(qr(l(x),l,r,L,M),qr(r(x),l,r,M+,R));
} int gt(int x,int ln) {for(int i=;~i;i--) if(l[fa[x][i]]>=ln) x=fa[x][i]; return x;}
void dfs(int x) {for(int i=hd[x];i;i=nx[i]) dfs(to[i]),rt[x]=mrg(rt[x],rt[to[i]]),fa[to[i]][]=x;} int main() {
scanf("%s%d",sr,&m);
for(int i=;i<=m;i++) {
scanf("%s",s),ls=;
for(int j=;s[j];j++) ins(s[j]-'a'),upd(rt[ls],i);
}
for(int i=,u=,s=;sr[i];i++) {
int c=sr[i]-'a';
while(u&&!ch[u][c]) u=f[u],s=l[u];
if(ch[u][c]) ln[i+]=++s,u=ch[u][c],po[i+]=u;
}
for(int i=;i<=sz;i++) ad(f[i],i);
dfs(),scanf("%d",&q);
for(int j=;j<;j++) for(int i=;i<=sz;i++) fa[i][j]=fa[fa[i][j-]][j-];
while(q--) {
scanf("%d%d%d%d",&l1,&r1,&l2,&r2);
if(ln[r2]<r2-l2+) {printf("%d 0\n",l1); continue;}
pr k=qr(rt[gt(po[r2],r2-l2+)],l1,r1);
if(k.first) printf("%d %d\n",-k.second,k.first); else printf("%d 0\n",l1);
}
return ;
}

Codeforces 666E Forensic Examination的更多相关文章

  1. Codeforces 666E Forensic Examination SAM or SA+线段树合并

    E. Forensic Examination http://codeforces.com/problemset/problem/666/E 题目大意:给模式串S以及m个特殊串,q个询问,询问S的子串 ...

  2. Codeforces.666E.Forensic Examination(广义后缀自动机 线段树合并)

    题目链接 \(Description\) 给定串\(S\)和\(m\)个串\(T_i\).\(Q\)次询问,每次询问\(l,r,p_l,p_r\),求\(S[p_l\sim p_r]\)在\(T_l\ ...

  3. Codeforces 666E Forensic Examination SAM+权值线段树

    第一次做这种$SAM$带权值线段树合并的题 然而$zjq$神犇看完题一顿狂码就做出来了 $Orz$ 首先把所有串当成一个串建$SAM$ 我们对$SAM$上每个点 建一棵权值线段树 每个叶子节点表示一个 ...

  4. codeforces 666E. Forensic Examination(广义后缀自动机,Parent树,线段树合并)

    传送门: 解题思路: 很坑的一道题,需要离线处理,假如只有一组询问,那么就可以直接将endpos集合直接累加输出就好了. 这里就要将询问挂在树节点上,在进行线段树合并时查询就好了. 代码超级容易写挂的 ...

  5. CodeForces - 666E: Forensic Examination (广义SAM 线段树合并)

    题意:给定字符串S,然后M个字符串T.Q次询问,每次给出(L,R,l,r),问S[l,r]在L到R这些T字符串中,在哪个串出现最多,以及次数. 思路:把所有串建立SAM,然后可以通过倍增走到[l,r] ...

  6. Codeforces 666E Forensic Examination(广义后缀自动机+线段树合并)

    将所有串(包括S)放一块建SAM.对于询问,倍增定位出该子串所在节点,然后要查询的就是该子串在区间内的哪个字符串出现最多.可以线段树合并求出该节点在每个字符串中的出现次数. #include<b ...

  7. CF 666E Forensic Examination 【SAM 倍增 线段树合并】

    CF 666E Forensic Examination 题意: 给出一个串\(s\)和\(n\)个串\(t_i\),\(q\)次询问,每次询问串\(s\)的子串\(s[p_l:p_r]\)在串\(t ...

  8. CF 666E Forensic Examination——广义后缀自动机+线段树合并

    题目:http://codeforces.com/contest/666/problem/E 对模式串建广义后缀自动机,询问的时候把询问子串对应到广义后缀自动机的节点上,就处理了“区间”询问. 还要处 ...

  9. [CF 666E] Forensic Examination

    Description 传送门 Solution 对 \(T[1..m]\) 建立广义后缀自动机,离线,找出代表 \(S[pl,pr]\) 的每个节点,线段树合并. Code #include < ...

随机推荐

  1. 【技巧】Java工程中的Debug信息分级输出接口

    也许本文的标题你们没咋看懂.但是,本文将带大家领略输出调试的威力. 灵感来源 说到灵感,其实是源于笔者在修复服务器的ssh故障时的一个发现. 这个学期初,同袍(容我来一波广告产品页面,同袍官网)原服务 ...

  2. C++ 实现一个信号量

    C++ 实现一个信号量 信号量有很多应用场景,事实上只要是生产者-消费者模型,一般都需要一个信号量来控制. POSIX接口是有PV信号量API的.但C++标准没有.下面是一个PV信号量的简单实现.有些 ...

  3. vue组件详解(二)——使用props传递数据

    在 Vue 中,父子组件的关系可以总结为 props向下传递,事件向上传递.父组件通过 props 给子组件下发数据,子组件通过事件给父组件发送消息.看看它们是怎么工作的.  一.基本用法 组件不仅仅 ...

  4. 深入了解GOT,PLT和动态链接

    之前几篇介绍exploit的文章, 有提到return-to-plt的技术. 当时只简单介绍了 GOT和PLT表的基本作用和他们之间的关系, 所以今天就来详细分析下其具体的工作过程. 本文所用的依然是 ...

  5. javascript学习(1)用户的Javascript 放在哪里和函数的绑定方式

    一.实验 1:js脚本放在那里最合适? 1.代码 1.1.test.html <!DOCTYPE html><html>    <head>        < ...

  6. IDE-Android Studio -FAQ-使用习惯(不断更新 欢迎留言)

    摘要: 从ecplise工具切换到android studio后遇到了很多问题,起初亦非常痛苦,城墙内外阅博无数才得以解决.所以把当时遇到的问题记录下来,方便后来人学习. 另如果有遇到未纪录的问题欢迎 ...

  7. kafka--producer 发布消息

    1. 写入方式 producer 采用 push 模式将消息发布到 broker,每条消息都被 append 到 patition 中,属于顺序写磁盘(顺序写磁盘效率比随机写内存要高,保障 kafka ...

  8. mysql(3)—— 内连接、外连接的区别

    先来看一下,内连接的语法: SELECT  XXX FROM XXX INNER JOIN XXX ON XXX; 这里 INNER 可以省略,在上一篇博客中我们对于笛卡尔积现象的研究中(http:/ ...

  9. POJ-1679 The Unique MST---判断最小生成树是否唯一

    题目链接: https://vjudge.net/problem/POJ-1679 题目大意: 给定一个无向连通网,判断最小生成树是否唯一. 思路: (1)对图中的每条边,扫描其他边,如果存在相同权值 ...

  10. 南阳OJ-12-喷水装置(二)贪心+区间覆盖

    题目链接: http://acm.nyist.edu.cn/JudgeOnline/problem.php?pid=12 题目大意: 有一块草坪,横向长w,纵向长为h,在它的橫向中心线上不同位置处装有 ...