题目

首先先对\(s\)建一个\(\operatorname{SAM}\),设\(w=kq\)

发现\(k,q\leq 10^5\),但是\(w\leq 10^5\),于是套路地根号讨论一下

如果\(k\leq \sqrt{w}\),那么说明单个询问串都很短,我们完全可以暴力长度为\(k\)的字符串的所有子串,之后二分查询一下这个子串在\([a,b]\)的询问里出现的次数,这个子串在\(s\)出现的次数直接边走边查就好了,这边的复杂度是\(O(qk^2\log m)=O(w\sqrt{w}\log m)\)

如果\(k>\sqrt{w}\),那么说明单个询问串很长,但是询问个数\(q\leq \sqrt{w}\),于是我们直接扫\([a,b]\)范围内的所有区间,考虑到子串\([l,r]\)在\(s\)中出现了多少次可以转化为\(s\)有多少个前缀和前缀\(r\)的\(\operatorname{LCS}\)长度不小于\(r-l+1\),于是直接树上倍增求解,复杂度\(O(qm\log n)=O(m\sqrt{w}\log n)\)

代码

#include<bits/stdc++.h>
#define re register
#define LL long long
inline int read() {
char c=getchar();int x=0;while(c<'0'||c>'9') c=getchar();
while(c>='0'&&c<='9') x=(x<<3)+(x<<1)+c-48,c=getchar();return x;
}
const int maxn=2e5+5;
int fa[maxn],A[maxn],tax[maxn],son[maxn][26],len[maxn],sz[maxn];
int n,m,k,q,lst,cnt;
char S[maxn>>1];
inline void ins(int c) {
int p=++cnt,f=lst;lst=p;
len[p]=len[f]+1,sz[p]=1;
while(f&&!son[f][c]) son[f][c]=p,f=fa[f];
if(!f) {fa[p]=1;return;}
int x=son[f][c];
if(len[f]+1==len[x]) {fa[p]=x;return;}
int y=++cnt;
len[y]=len[f]+1,fa[y]=fa[x],fa[p]=fa[x]=y;
for(re int i=0;i<26;i++) son[y][i]=son[x][i];
while(f&&son[f][c]==x) son[f][c]=y,f=fa[f];
}
namespace sub1 {
int id[350][350],tot;
std::vector<int> v[maxn>>1];
inline int find(int t,int x) {
int l=0,r=v[t].size()-1,h=-1;
while(l<=r) {
int mid=l+r>>1;
if(v[t][mid]<=x) h=mid,l=mid+1;
else r=mid-1;
}
return h+1;
}
void solve() {
for(re int i=1;i<=k;++i)
for(re int j=i;j<=k;++j) id[i][j]=++tot;
for(re int x,y,i=1;i<=m;i++) {
x=read()+1,y=read()+1;
v[id[x][y]].push_back(i);
}
for(re int x,y,i=1;i<=q;i++) {
scanf("%s",S+1);x=read(),y=read()+1;
LL ans=0;
for(re int now=1,j=1;j<=k;++j,now=1)
for(re int p=j;p<=k;++p) {
now=son[now][S[p]-'a'];
if(!now) break;
ans+=1ll*sz[now]*(find(id[j][p],y)-find(id[j][p],x));
}
printf("%lld\n",ans);
}
}
}
namespace sub2 {
int ql[maxn>>1],qr[maxn>>1],deep[maxn];
int lg[maxn>>1],pos[maxn>>1],ml[maxn>>1],f[18][maxn];
void solve() {
for(re int i=1;i<=m;i++) ql[i]=read()+1,qr[i]=read()+1;
deep[1]=1;for(re int i=2;i<=n;i++) lg[i]=lg[i>>1]+1;
for(re int i=2;i<=cnt;i++) deep[A[i]]=deep[fa[A[i]]]+1;
for(re int i=1;i<=cnt;i++) {
int x=A[i];f[0][x]=fa[x];
for(re int j=1;j<=lg[deep[x]];++j)
f[j][x]=f[j-1][f[j-1][x]];
}
int x,y;
while(q--) {
scanf("%s",S+1);x=read()+1,y=read()+1;
int now=1,l=0;LL ans=0;
for(re int i=1;i<=k;i++) {
if(son[now][S[i]-'a']) {
pos[i]=now=son[now][S[i]-'a'];
ml[i]=++l;continue;
}
while(now&&!son[now][S[i]-'a']) now=fa[now];
if(!now) pos[i]=now=1,ml[i]=l=0;
else ml[i]=l=len[now]+1,pos[i]=now=son[now][S[i]-'a'];
}
for(re int i=x;i<=y;++i) {
if(ml[qr[i]]<qr[i]-ql[i]+1) continue;
now=pos[qr[i]];
for(re int j=lg[deep[now]];j>=0;--j)
if(len[f[j][now]]>=qr[i]-ql[i]+1) now=f[j][now];
ans+=sz[now];
}
printf("%lld\n",ans);
}
}
}
int main() {
n=read(),m=read(),q=read(),k=read();
scanf("%s",S+1);lst=cnt=1;
for(re int i=1;i<=n;i++) ins(S[i]-'a');
for(re int i=1;i<=cnt;i++) tax[len[i]]++;
for(re int i=1;i<=n;i++) tax[i]+=tax[i-1];
for(re int i=1;i<=cnt;i++) A[tax[len[i]]--]=i;
for(re int i=cnt;i;--i) sz[fa[A[i]]]+=sz[A[i]];
if(k<=std::sqrt(k*q)) sub1::solve();else sub2::solve();
return 0;
}

loj6031「雅礼集训 2017 Day1」字符串的更多相关文章

  1. 并不对劲的Loj6031:「雅礼集训 2017 Day1」字符串

    题目传送门:-> 看到题目的第一反应当然是暴力:对于串s建后缀自动机,每次询问中,求w对应的子串在s的SAM中的right集合.O(qmk)听上去显然过不了. 数据范围有个∑w<=1e5, ...

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

    [LOJ 6031] 「雅礼集训 2017 Day1」字符串 题意 给定一个长度为 \(n\) 的字符串 \(s\), \(m\) 对 \((l_i,r_i)\), 回答 \(q\) 个询问. 每个询 ...

  3. loj#6031. 「雅礼集训 2017 Day1」字符串(SAM 广义SAM 数据分治)

    题意 链接 Sol \(10^5\)次询问每次询问\(10^5\)个区间..这种题第一感觉就是根号/数据分治的模型. \(K\)是个定值这个很关键. 考虑\(K\)比较小的情况,可以直接暴力建SAM, ...

  4. 「雅礼集训 2017 Day1」字符串 SAM、根号分治

    LOJ 注意到\(qk \leq 10^5\),我们很不自然地考虑根号分治: 当\(k > \sqrt{10^5}\),此时\(q\)比较小,与\(qm\)相关的算法比较适合.对串\(s\)建S ...

  5. loj 6031「雅礼集训 2017 Day1」字符串

    loj 注意到每次询问串长度都是给定的,并且询问串长\(k*\)询问次数\(q<10^5\),所以这里面一个东西大的时候另一个东西就小,那么考虑对较小的下功夫 如果\(k\le \sqrt{n} ...

  6. 「雅礼集训 2017 Day1」 解题报告

    「雅礼集训 2017 Day1」市场 挺神仙的一题.涉及区间加.区间除.区间最小值和区间和.虽然标算就是暴力,但是复杂度是有保证的. 我们知道如果线段树上的一个结点,\(max=min\) 或者 \( ...

  7. [LOJ 6030]「雅礼集训 2017 Day1」矩阵

    [LOJ 6030] 「雅礼集训 2017 Day1」矩阵 题意 给定一个 \(n\times n\) 的 01 矩阵, 每次操作可以将一行转置后赋值给某一列, 问最少几次操作能让矩阵全为 1. 无解 ...

  8. [LOJ 6029]「雅礼集训 2017 Day1」市场

    [LOJ 6029] 「雅礼集训 2017 Day1」市场 题意 给定一个长度为 \(n\) 的数列(从 \(0\) 开始标号), 要求执行 \(q\) 次操作, 每次操作为如下四种操作之一: 1 l ...

  9. loj#6030. 「雅礼集训 2017 Day1」矩阵(贪心 构造)

    题意 链接 Sol 自己都不知道自己怎么做出来的系列 不难观察出几个性质: 最优策略一定是先把某一行弄黑,然后再用这一行去覆盖不是全黑的列 无解当且仅当无黑色.否则第一个黑色所在的行\(i\)可以先把 ...

随机推荐

  1. Jmeter-【JSON Extractor】-响应结果中三级key取值

    一.请求返回样式 二.取第三个option 三.查看结果

  2. thinkphp 模型调试

    调试执行的SQL语句 在模型操作中 ,为了更好的查明错误,经常需要查看下最近使用的SQL语句,我们可以用getLastsql方法来输出上次执行的sql语句.例如: $User = M("Us ...

  3. 线性dp——hdu6578经典dp

    多校第一场第一题,这种类型的dp之前做过两题,状态转移一般是从当前状态往后推的 很经典的dp,不过很卡时间 /* 定义 dp[t][i][j][k]代表填完前 t 个位置后,{0, 1, 2, 3} ...

  4. 线性dp——cf988F

    不是很难,dp[i]表示到位置i的最小花费 #include<bits/stdc++.h> using namespace std; #define ll long long #defin ...

  5. win7+vs2010配置驱动开发环境(问题种种版...)

     本来按照这个来做,能跑通helloworld,可是复杂的驱动就会出错....不知道什么原因,后来就直接用命令行来编译的. -------------------------------------- ...

  6. RouterOS视频教程下载

    下载信息 名称:RouterOS视频教程下载 格式:MP4 版本:V1.0 https://pan.baidu.com/s/1skU6oW1 下载密码:nb97

  7. C不同变量类型存储大小引发的BUG

    #include"stdio.h" typedef signed char int8; typedef unsigned char uint8; typedef signed sh ...

  8. solr 查询同一个core 的关联字段

    实现一个core里面多个字段的关联查询: 应用场景是: 词, 句子,文章 希望通过查询实现词,句子,文章里面共同有的关键字 private static CloudSolrServer cloudSo ...

  9. java执行spark查询hbase的jar包出现错误提示:ob aborted due to stage failure: Master removed our application: FAILED

    执行java调用scala 打包后的jar时候出现异常 /14 23:57:08 WARN TaskSchedulerImpl: Initial job has not accepted any re ...

  10. ps-手捧城堡滴水云雾图

    1打开背景图 置入第二张图片 栅格化-加入蒙版-渐变 置入第三张图片 栅格化-用快速选择工具选取-加入蒙版 置入第四张图片 栅格化-调整图层-点击城堡建立蒙版-点击手的蒙版 ctrl-点击城堡的蒙版- ...