题目

有趣的思想

首先暴力的话,自然是对每一个询问在\(AC\)自动机上跑一遍\(k\),看看跑出来的节点在\(fail\)树到根的路径上有多少个\(l\)到\(r\)之间的结束标记就好了

我们发现无论怎么优化好像都不是很可行,考虑一下对根号优化

对于长度大于\(\sqrt{n}\)的串,显然这样的串也不会超过\(\sqrt{n}\)个,我们把这些串在\(AC\)机上跑一遍,之后统计一下子树和,统计一个前缀和就可以回答询问了

这样复杂度是\(O(n\sqrt{n})\)

对于长度小于\(\sqrt{n}\)的串,我们允许把每一个串都放到\(AC\)自动机上跑一遍,于是可以直接用主席树来查一下跑出来的节点到根有多少个标记在\(l\)到\(r\)之间

这边的复杂度是\(O(n\sqrt{n}logn)\)

代码

#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<string>
#include<vector>
#include<cmath>
#include<queue>
#define maxn 100005
#define M 4000005
#define re register
#define LL long long
#define int long long
#define max(a,b) ((a)>(b)?(a):(b))
#define min(a,b) ((a)<(b)?(a):(b))
inline int read() {
int x=0;char c=getchar();while(c<'0'||c>'9') c=getchar();
while(c>='0'&&c<='9') x=(x<<3)+(x<<1)+c-48,c=getchar();return x;
}
struct E{int v,nxt;}e[maxn];
struct Ask{int l,r,k,rk,o;}q[maxn];
int n,m,cnt,tot,num,sz;
int fail[maxn],son[maxn][26],len[maxn],head[maxn],a[maxn];
LL sum[maxn],pre[maxn],Ans[maxn];
int rt[maxn],ls[M],rs[M],d[M];
std::string S[maxn];
char T[maxn];
std::vector<int> v[maxn];
inline int cmp1(Ask A,Ask B) {if(A.o==B.o)return A.k<B.k;return A.o>B.o;}
inline void add(int x,int y) {e[++num].v=y;e[num].nxt=head[x];head[x]=num;}
inline void work() {for(re int i=1;i<=n;i++) pre[i]=pre[i-1]+sum[a[i]];}
inline void ins(int i) {
scanf("%s",T);
S[i]=T;len[i]=S[i].size();tot+=len[i];
int now=0;
for(re int j=0;j<S[i].size();j++) {
if(!son[now][S[i][j]-'a']) son[now][S[i][j]-'a']=++cnt;
now=son[now][S[i][j]-'a'];
}
v[now].push_back(i);a[i]=now;
}
inline void Build() {
std::queue<int> q;
for(re int i=0;i<26;i++) if(son[0][i]) q.push(son[0][i]);
while(!q.empty()) {
int k=q.front();q.pop();
for(re int i=0;i<26;i++)
if(son[k][i]) fail[son[k][i]]=son[fail[k]][i],q.push(son[k][i]);
else son[k][i]=son[fail[k]][i];
}
}
inline int change(int pre,int x,int y,int pos) {
int root=++sz;
d[root]=d[pre]+1;
if(x==y) return root;
ls[root]=ls[pre],rs[root]=rs[pre];
int mid=x+y>>1;
if(pos<=mid) ls[root]=change(ls[pre],x,mid,pos);
else rs[root]=change(rs[pre],mid+1,y,pos);
return root;
}
inline int query(int p,int x,int y,int pos) {
if(!pos) return 0;
if(x==y) return d[p];
int mid=x+y>>1;
if(pos<=mid) return query(ls[p],x,mid,pos);
return d[ls[p]]+query(rs[p],mid+1,y,pos);
}
inline void Query(int i) {
int now=0;
for(re int j=0;j<len[i];j++)
now=son[now][S[i][j]-'a'],sum[now]++;
}
inline void dfs(int x) {
for(re int i=head[x];i;i=e[i].nxt) {
dfs(e[i].v),sum[x]+=sum[e[i].v];
}
}
inline void Dfs(int x) {
for(re int i=head[x];i;i=e[i].nxt) {
rt[e[i].v]=rt[x];
for(re int j=0;j<v[e[i].v].size();j++) rt[e[i].v]=change(rt[e[i].v],1,n,v[e[i].v][j]);
Dfs(e[i].v);
}
}
signed main() {
n=read(),m=read();
for(re int i=1;i<=n;i++) ins(i);
tot=std::sqrt(tot);Build();
for(re int i=1;i<=m;i++)
q[i].l=read(),q[i].r=read(),q[i].k=read(),q[i].rk=i,q[i].o=(len[q[i].k]>tot);
std::sort(q+1,q+m+1,cmp1);
for(re int i=1;i<=cnt;i++) add(fail[i],i);
Dfs(0);
int L=1,R=1;Query(q[1].k);dfs(0),work();
for(re int i=2;i<=m+1;i++) {
if(len[q[i].k]<=tot) {for(re int j=L;j<i;j++) Ans[q[j].rk]=pre[q[j].r]-pre[q[j].l-1];R=i;break;}
if(q[i].k!=q[i-1].k) {
for(re int j=L;j<i;j++) Ans[q[j].rk]=pre[q[j].r]-pre[q[j].l-1];
memset(sum,0,sizeof(sum));
Query(q[i].k);dfs(0);work();L=i;
}
}
for(re int i=R;i<=m;i++) {
int now=0;
for(re int j=0;j<len[q[i].k];j++)
now=son[now][S[q[i].k][j]-'a'],Ans[q[i].rk]+=query(rt[now],1,n,q[i].r)-query(rt[now],1,n,q[i].l-1);
}
for(re int i=1;i<=m;i++) printf("%I64d\n",Ans[i]);
return 0;
}

CF587F Duff is Mad的更多相关文章

  1. [CF587F]Duff is Mad[AC自动机+根号分治+分块]

    题意 给你 \(n\) 个串 \(s_{1\cdots n}\) ,每次询问给出 \(l,r,k\) ,问在 \(s_{l\cdots r}\) 中出现了多少次 \(s_k\) . \(n,q,\su ...

  2. CF587F Duff is Mad(AC自动机+树状数组+分块)

    考虑两一个暴力 1 因为询问\([a,b]\)可以拆成\([1,b]\)-\([1,a-1]\)所以把询问离线,然后就是求\([1,x]\)中被\(S_i\)包含的串的数量.考虑当\([1,x-1]- ...

  3. 【CF587F】Duff is Mad AC自动机+分块

    [CF587F]Duff is Mad 题意:给出n个串$s_1,s_2..s_n$,有q组询问,每次给出l,r,k,问你编号在[l,r]中的所有串在$s_k$中出现了多少次. $\sum|s_i|, ...

  4. Codeforces 587F - Duff is Mad(根号分治+AC 自动机+树状数组)

    题面传送门 第一眼看成了 CF547E-- 话说 CF587F 和 CF547E 出题人一样欸--还有另一道 AC 自动机的题 CF696D 也是这位名叫 PrinceOfPersia 的出题人出的- ...

  5. CF数据结构练习

    1. CF 438D The Child and Sequence 大意: n元素序列, m个操作: 1,询问区间和. 2,区间对m取模. 3,单点修改 维护最大值, 取模时暴力对所有>m的数取 ...

  6. cf Round 587

    A.Duff and Weight Lifting(思维) 显然题目中只有一种情况可以合并 2^a+2^a=2^(a+1).我们把给出的mi排序一下,模拟合并操作即可. # include <c ...

  7. Lesson 21 Mad or not?

    Text Aeroplanes are slowly driving me mad. I live near an airport and passing planes can be heard ni ...

  8. Codeforces Round #326 (Div. 2) B. Pasha and Phone C. Duff and Weight Lifting

    B. Pasha and PhonePasha has recently bought a new phone jPager and started adding his friends' phone ...

  9. 【转】Duff's Device

    在看strcpy.memcpy等的实现发现用了内存对齐,每一个word拷贝一次的办法大大提高了实现效率,参加该blog(http://totoxian.iteye.com/blog/1220273). ...

随机推荐

  1. bzoj 2164: 采矿

    Description 浩浩荡荡的cg大军发现了一座矿产资源极其丰富的城市,他们打算在这座城市实施新的采矿战略.这个城市可以看成一棵有n个节点的有根树,我们把每个节点用1到n的整数编号.为了方便起见, ...

  2. apache配置apache server status,监控服务器访问情况

    在apache配置文件中添加开启代码, 1.如果你的Apache配置文件httpd.conf或者extra/httpd-info.conf中有LoadModule status_module modu ...

  3. js定义一个处理字符串的函数

    //定义一个处理字符串的方法 function StringBuffer(str){ var arr = []; str = str || ''; arr.push(str); //追加字符串 thi ...

  4. Nginx - 简易图片服务器

    安装 主要使用Nginx和vsftpd. 安装方面可以直接从nginx官网上下载,或者... yum install nginx 如果没有yum源则需要自行添加再进行install. yum inst ...

  5. php strpos返回字符串首次出现的位置

    (PHP 4, PHP 5, PHP 7) strpos — 查找字符串首次出现的位置 说明 mixed strpos ( string $haystack , mixed $needle [, in ...

  6. Filter---javaweb的过滤器

    1.Filter是什么? Filter的基本功能是对Servlet容器调用Servlet的过程进行拦截,从而在Servlet进行响应处理的前后实现一些特殊的功能. 在Servlet API中定义了三个 ...

  7. hdu 1011 树形背包

    http://blog.csdn.net/libin56842/article/details/9876503 这道题和poj 1155的区别是: poj1155是边的价值,所以从边的关系入手 hdu ...

  8. nginx: [emerg] bind() to 0.0.0.0:80 failed (98: Address already in use)解决方案

    前提:已经配置好静态IP以防万一,先安装好iptables服务(不管你装没装,先执行,免得后面添乱)[root@localhost ~]# yum install iptables-services[ ...

  9. Spring 框架(一)

    1 spring框架概述 1.1 什么是spring l Spring是一个开源框架,Spring是于2003 年兴起的一个轻量级的Java 开发框架,由Rod Johnson 在其著作Expert ...

  10. Java基础学习总结一(Java语言发展历史、JDK下载安装以及配置环境变量)

    最近一段时间计划复习一下java基础知识,使用的视频课程是尚学堂高淇老师的,上课过程中的心得体会直接总结一下,方便以后复习. 一:计算机语言的发展 1:机器语言,最原始的语言,主要有“01”构成,最早 ...