[LOJ 6031] 「雅礼集训 2017 Day1」字符串

题意

给定一个长度为 \(n\) 的字符串 \(s\), \(m\) 对 \((l_i,r_i)\), 回答 \(q\) 个询问. 每个询问会给定一个长度为 \(k\) 的字符串 \(w\) 以及一对 \(L,R\), 求所有满足 \(i\in [L,R]\) 的 \(w[l_i:r_i]\) 在 \(s\) 中的出现次数之和.

\(n,m,k,q\le 1\times 10^5\), \(\sum |w|\le 1\times 10^5\).

题解

这sb题直接SAM暴力tm有90分...然而考场上把 \(k\) 和 \(q\) 读反了挂成10分...

那个 \(\sum |w|\) 显然就是 \(kq\), 于是 \(kq\le1\times 10^5\). 我们一点都不自然地想到可以根号分类.

当 \(k\le \sqrt m\) 的时候, 显然 \(k^2q=O(m)k=O(m\sqrt m)\). 那么我们直接枚举 \(w\) 的所有子串, 在 \(s\) 的SAM上倍增计算答案, 二分计算当前查询区间中有多少个当前子串就可以统计当前子串对答案的贡献了. 时间复杂度是 \(O(k^2q\log n)=O(m\sqrt m\log n)\).

当 \(k>\sqrt m\) 的时候, 显然 \(kq=O(m)\Rightarrow q=O(\sqrt m)\). 于是我们将所有 \((l_i,r_i)\) 分布到 \(r_i\) 上, 然后边在SAM上跑边计算右端点为当前位置的子串对答案的贡献. 一次的复杂度是 \(O(k+m\log n)\). 总复杂度显然是 \(O(m\sqrt m \log n)\).

然后就过了...

参考代码

#include <bits/stdc++.h>

namespace rvalue{
const int MAXN=2e5+10;
typedef long long intEx; int n;
int q;
int k;
int m;
int lg;
int cnt=1;
int last=1;
int root=1;
int s[MAXN];
int prt[MAXN];
int len[MAXN];
int size[MAXN];
char buf[MAXN];
int pprt[20][MAXN];
std::map<char,int> chd[MAXN]; void Extend(char); namespace BF1{
const int SQRN=350; std::vector<int> qpos[SQRN][SQRN]; int main(){
#ifdef IRECT
puts("BF1");
#endif
for(int i=0;i<m;i++){
int l,r;
scanf("%d%d",&l,&r);
qpos[l][r].push_back(i);
}
while(q--){
int a,b;
scanf("%s%d%d",buf,&a,&b);
int cur=root,curlen=0;
intEx ans=0;
for(int i=0;i<k;i++){
while(cur!=root&&!chd[cur].count(buf[i])){
cur=prt[cur];
curlen=std::min(curlen,len[cur]);
}
if(chd[cur].count(buf[i])){
cur=chd[cur][buf[i]];
++curlen;
int pos=cur;
for(int plen=curlen;plen>=1;plen--){
while(len[prt[pos]]>=plen)
pos=prt[pos];
auto& q=qpos[i-plen+1][i];
auto low=std::lower_bound(q.begin(),q.end(),a);
auto up=std::upper_bound(q.begin(),q.end(),b);
int cnt=up-low;
// printf("%d len=%d cnt=%d\n",i,plen,cnt);
ans+=1ll*cnt*size[pos];
}
}
}
printf("%lld\n",ans);
}
return 0;
}
} namespace BF2{
std::vector<std::pair<int,int>> qlen[MAXN]; int main(){
#ifdef IRECT
puts("BF2");
#endif
for(int i=0;i<m;i++){
int l,r;
scanf("%d%d",&l,&r);
qlen[r].emplace_back(r-l+1,i);
}
while(q--){
int a,b;
scanf("%s%d%d",buf,&a,&b);
int cur=root,curlen=0;
intEx ans=0;
for(int i=0;i<k;i++){
while(cur!=root&&!chd[cur].count(buf[i])){
cur=prt[cur];
curlen=std::min(curlen,len[cur]);
}
if(chd[cur].count(buf[i])){
cur=chd[cur][buf[i]];
++curlen;
for(auto q:qlen[i]){
if(q.second<a||q.second>b)
continue;
if(curlen<q.first)
continue;
int pos=cur;
for(int j=lg;j>=0;j--)
if(len[pprt[j][pos]]>=q.first)
pos=pprt[j][pos];
ans+=size[pos];
}
}
}
printf("%lld\n",ans);
}
return 0;
}
} int main(){
scanf("%d%d%d%d",&n,&m,&q,&k);
scanf("%s",buf);
for(int i=0;i<n;i++)
Extend(buf[i]);
for(int i=1;i<=cnt;i++)
s[i]=i;
std::sort(s+1,s+cnt+1,[](int a,int b){return len[a]>len[b];});
for(int i=1;i<=cnt;i++){
size[prt[s[i]]]+=size[s[i]];
pprt[0][i]=prt[i];
}
for(int j=1;(1<<j)<=cnt;j++){
lg=j;
for(int i=1;i<=cnt;i++)
pprt[j][i]=pprt[j-1][pprt[j-1][i]];
}
if(1ll*k*k<=m)
BF1::main();
else
BF2::main();
return 0;
} void Extend(char x){
int p=last;
int np=++cnt;
size[last=np]=1;
len[np]=len[p]+1;
while(p&&!chd[p].count(x))
chd[p][x]=np,p=prt[p];
if(!p)
prt[np]=root;
else{
int q=chd[p][x];
if(len[q]==len[p]+1)
prt[np]=q;
else{
int nq=++cnt;
len[nq]=len[p]+1;
chd[nq]=chd[q];
prt[nq]=prt[q];
prt[q]=nq;
prt[np]=nq;
while(p&&chd[p][x]==q)
chd[p][x]=nq,p=prt[p];
}
}
}
} int main(){
#if 0
freopen("string.in","r",stdin);
freopen("string.out","w",stdout);
#endif
rvalue::main();
return 0;
}

最新文章

  1. sonn_game网站开发01:写在最前面
  2. day--6_python常用模块
  3. JNI输出log信息
  4. 六、动态增加方法Category
  5. Oracle数据库—— 事务处理与并发控制
  6. 国产神通数据库操作备忘(Linux)
  7. cloudera manager 及CDH卸载
  8. Form表单中的三种查询方法
  9. python 发邮件
  10. 什麼是 N-key 與按鍵衝突?原理說明、改善技術、選購注意完全解析
  11. Highways(求最小生成树的最大边)
  12. Java的基本语法
  13. 洛谷3月月赛 R1 Step! ZERO to ONE
  14. Java内存模型JMM 高并发原子性可见性有序性简介 多线程中篇(十)
  15. monkeyrunner.bat运行python脚本/命令行
  16. swagger 指定字段不显示到文档里
  17. A1127. ZigZagging on a Tree
  18. Zookeeper系列五:Master选举、ZK高级特性:基本模型
  19. 通用程序返回结果类 ApplicationResult.cs
  20. 轻量级Java持久化框架,Hibernate完美助手,Minidao 1.6.2版本发布

热门文章

  1. 使用Unicode字符实现换行
  2. HDU 1242 Rescue(BFS+优先队列)
  3. SQL SERVER存储过程中使用事务
  4. Thinkphp 图片上传
  5. 【游记】Noip2018
  6. c# 对文件的各种操作
  7. oracle逐步学习总结之权限和角色(基础六)
  8. python学习之老男孩python全栈第九期_day011作业
  9. python的变量以及常量介绍
  10. Angular 6.X CLI(Angular.json) 属性详解