题面

CmdOI2019 口头禅

给 \(n\) 个 \(01\) 串 \(s_i\),\(m\) 个询问问 \(s_{l\sim r}\) 的最长公共子串长度。

数据范围:\(1\le n\le 20000\),\(1\le m\le 10^5\),\(\sum |s_i|\le 4\cdot 10^5\)。


蒟蒻语

蒟蒻看到这个题口胡了一个做法,然后轻松拿到了最优解,发现这是道大水题。

什么猫树或分治我没听说过,反正广义 SAM 上枚举子串乱搞 \(\Theta(n\sqrt n\log n)\) 跑得飞快。


蒟蒻解

首先把串建成广义 SAM 没有问题,为了方便我写了盗版的 /ch

离线询问,把 \([l_i,r_i]\) 这个询问挂到 \(r_i\) 上。

顺序枚举 \(r\),同时干这些坏事:

把 \(r_i=r\) 的询问按 \(l_i\) 排序,设有 \(qn\) 个询问。

对于 SAM 的节点 \(p\),维护 \(li_p\) 和 \(ri_p\),表示 \(s_{1\sim r}\) 中这个以这个节点为子串的最右连续区间。

维护方法是枚举 \(s_r\) 的所有子串(暴力跳每个前缀的 \(fa\),时间复杂度 \(\Theta(n\sqrt n)\)),通过 \(r-1\) 递推。

然后对于每个该串子串节点 \(p\),lower_bound 找到第一个 \(l_i\ge li_p\),对询问 \([i,qn]\) 的答案都与该节点代表最长串长度取 \(\max\)。

这东西根据单调性差分一下即可,然后就做完了。


代码

#include <bits/stdc++.h>
using namespace std; //Start
typedef long long ll;
typedef double db;
#define mp(a,b) make_pair((a),(b))
#define x first
#define y second
#define be(a) (a).begin()
#define en(a) (a).end()
#define sz(a) int((a).size())
#define pb(a) push_back(a)
#define R(i,a,b) for(int i=(a),I=(b);i<I;i++)
#define L(i,a,b) for(int i=(b)-1,I=(a)-1;i>I;i--)
const int iinf=0x3f3f3f3f;
const ll linf=0x3f3f3f3f3f3f3f3f; //Data
const int N=2e4,lN=4e5,qN=1e5;
int n,qn,ans[qN];
string s[N];
vector<pair<int,int>> que[N]; //SuffixAutoMoton
const int tN=(lN<<1)+1,cN=2;
int tn,ch[tN][cN],fa[tN],len[tN];
int newsam(){
fill(ch[tn],ch[tn]+cN,-1),fa[tn]=-1;
return tn++;
}
int rt=newsam(),t;
void extend(int c){
int p=t,np=t=newsam();
len[np]=len[p]+1;
for(;~p&&!~ch[p][c];p=fa[p]) ch[p][c]=np;
if(!~p) fa[np]=rt;
else {
int q=ch[p][c];
if(len[q]==len[p]+1) fa[np]=q;
else {
int nq=newsam();
copy(ch[q],ch[q]+cN,ch[nq]);
len[nq]=len[p]+1,fa[nq]=fa[q],fa[q]=fa[np]=nq;
for(;~p&&ch[p][c]==q;p=fa[p]) ch[p][c]=nq;
}
}
}
int li[tN],ri[tN],vis[tN]; //Main
int main(){
ios::sync_with_stdio(0);
cin.tie(0),cout.tie(0);
cin>>n>>qn;
R(i,0,n){
cin>>s[i],t=rt;
for(char c:s[i]) extend(c-'0');
}
R(i,0,qn){
int l,r; cin>>l>>r,--l,--r;
que[r].pb(mp(l,i));
}
fill(vis,vis+tn,-1);
fill(li,li+tn,-2),fill(ri,ri+tn,-2);
R(i,0,n){
sort(be(que[i]),en(que[i]));
vector<int> mx(sz(que[i])+1);
int p=rt,now=0;
for(char c:s[i]){
int q=p=ch[p][c-'0']; now++;
for(;~q&&vis[q]<i;q=fa[q]){
vis[q]=i;
if(ri[q]==i-1) ri[q]=i;
else li[q]=ri[q]=i;
int id=lower_bound(be(que[i]),en(que[i]),
mp(li[q],-1))-be(que[i]);
mx[id]=max(mx[id],min(len[q],now));
}
}
R(j,0,sz(que[i])) mx[j+1]=max(mx[j+1],mx[j]);
R(j,0,sz(que[i])) ans[que[i][j].y]=mx[j];
}
R(i,0,qn) cout<<ans[i]<<"\n";
return 0;
}

祝大家学习愉快!

题解-CmdOI2019 口头禅的更多相关文章

  1. 2016 华南师大ACM校赛 SCNUCPC 非官方题解

    我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...

  2. noip2016十连测题解

    以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...

  3. BZOJ-2561-最小生成树 题解(最小割)

    2561: 最小生成树(题解) Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1628  Solved: 786 传送门:http://www.lyd ...

  4. Codeforces Round #353 (Div. 2) ABCDE 题解 python

    Problems     # Name     A Infinite Sequence standard input/output 1 s, 256 MB    x3509 B Restoring P ...

  5. 哈尔滨理工大学ACM全国邀请赛(网络同步赛)题解

    题目链接 提交连接:http://acm-software.hrbust.edu.cn/problemset.php?page=5 1470-1482 只做出来四道比较水的题目,还需要加强中等题的训练 ...

  6. 2016ACM青岛区域赛题解

    A.Relic Discovery_hdu5982 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Jav ...

  7. poj1399 hoj1037 Direct Visibility 题解 (宽搜)

    http://poj.org/problem?id=1399 http://acm.hit.edu.cn/hoj/problem/view?id=1037 题意: 在一个最多200*200的minec ...

  8. 网络流n题 题解

    学会了网络流,就经常闲的没事儿刷网络流--于是乎来一发题解. 1. COGS2093 花园的守护之神 题意:给定一个带权无向图,问至少删除多少条边才能使得s-t最短路的长度变长. 用Dijkstra或 ...

  9. CF100965C题解..

    求方程 \[ \begin{array}\\ \sum_{i=1}^n x_i & \equiv & a_1 \pmod{p} \\ \sum_{i=1}^n x_i^2 & ...

随机推荐

  1. 《.NET 5.0 背锅案》第5集-案情大转弯:都是我们的错,让 .NET 5.0 背锅

    第1集:验证 .NET 5.0 正式版 docker 镜像问题 第2集:码中的小窟窿,背后的大坑,发现重要嫌犯 EnyimMemcachedCore 第3集-剧情反转:EnyimMemcachedCo ...

  2. 手写一个最迷你的Web服务器

    今天我们就仿照Tomcat服务器来手写一个最简单最迷你版的web服务器,仅供学习交流. 1. 在你windows系统盘的F盘下,创建一个文件夹webroot,用来存放前端代码.  2. 代码介绍: ( ...

  3. 关于多线程--网络编程 -- 注解反射的一点笔记(JAVA篇)

    一 . 线程 java开启一个线程的方法(三种) 方法一:继承Thread类并New一个线程对象 步骤: 1):定义一个类A继承于Java.lang.Thread类. class TestThread ...

  4. Python_异常处理、调试

    1.try except 机制 # 错误处理 # 一般程序都要用到错误捕获,当没有加且有错误的时候Python解释器会执行错误捕获,且是一层层向上捕获[所以问题点会在最下面] try: print(' ...

  5. git 强制更新本地和强制提交覆盖

    强制更新覆盖本地: git pull时出现冲突 放弃本地修改,使远程库内容强制覆盖本地代码git fetch --all //只是下载代码到本地,不进行合并操作git reset --hard ori ...

  6. Redis多线程原理详解

    本篇文章为你解答一下问题: 0:redis单线程的实现流程是怎样的? 1:redis哪些地方用到了多线程,哪些地方是单线程? 2:redis多线程是怎么实现的? 3:redis多线程是怎么做到无锁的? ...

  7. mongodb 副本集之入门篇

    作者: 凹凸曼-军军 前言:mongodb 因为高性能.高可用性.支持分片等特性,作为非关系型数据库被大家广泛使用.其高可用性主要是体现在 mongodb 的副本集上面(可以简单理解为一主多从的集群) ...

  8. centos xargs

    有些时候过滤后的东西需要传递给后面其它命令执行实现需求,这个时候xargs就派上用场了. 比如我想把过滤后的东东拷贝至其它目录,其实我可以进入那个目录然后执行ls,然后是过滤,接着再手工以拷贝那样也可 ...

  9. houdini 鱼眼相机

    http://mattebb.com/weblog/houdini-fisheye-camera/ 这个网站是有提供一个相机shader的,,如图是方形的,国内的用户,比较多是做球幕的小伙伴,圆形就行 ...

  10. JAVA在最新版Windows10_1909版本环境下的环境变量配置

    1.配置 1.1新建 JAVA_HOME C:\Program Files\Java\jdk-13.0.2 1.2新建 CLASSPATH .;%JAVA_HOME%\bin;%JAVA_HOME%\ ...