传送门

考虑一下,如果串B在串A中出现过,那么A的fail指针必定直接或间接指向B

那么我们可以把fail树建起来,那么就变成B代表的节点的子树里有多少节点属于A

然后这就是一个序列统计问题,直接用dfs序+树状数组可以维护

具体的操作就是,先把每一个点有关的询问给存起来,然后等到在trie树上一遍dfs,当dfs到这个串的结束节点说明所有有关这个串的节点都被遍历到(可以在遍历的时候顺便用树状数组维护序列和),然后对每一个询问查一下子树和即可

 //minamoto
#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
#include<algorithm>
using namespace std;
const int N=;
int tot=;
struct node{
int num,type;
node(){}
node(int num,int type):num(num),type(type){}
};
queue<node> q;
int ch[N][],End[N],word,fa[N],fail[N],dis[N];
inline int insert(int p,int c){
if(ch[p][c]) return ch[p][c];
ch[p][c]=++tot,fa[tot]=p;return tot;
}
inline int back(int p){return fa[p];}
inline void ed(int p){End[p]=++word,dis[word]=p;}
void build(){
for(int i=;i<;++i)
if(ch[][i])
q.push(node(ch[][i],i)),fail[ch[][i]]=;
while(!q.empty()){
node now=q.front();q.pop();
if(fail[now.num]!=){
int trail=fail[fa[now.num]];
while(true){
if(ch[trail][now.type]){trail=ch[trail][now.type];break;}
if(trail==) break;trail=fail[trail];
}
fail[now.num]=trail;
}
for(int i=;i<;++i)
if(ch[now.num][i])
q.push(node(ch[now.num][i],i));
}
}
int ver[N<<],Next[N<<],head[N],E=;
inline void add(int u,int v){
ver[++E]=v,Next[E]=head[u],head[u]=E;
}
int ver1[N],Next1[N],head1[N],edge1[N],E1=;
inline void add1(int u,int v,int num){
ver1[++E1]=v,Next1[E1]=head1[u],head1[u]=E1,edge1[E1]=num;
}
int c[N<<];
inline void addd(int x,int y){
for(;x<=tot;x+=x&-x) c[x]+=y;
}
inline int sum(int x){
int res=;
for(;x;x-=x&-x) res+=c[x];
return res;
}
int dfn[N],sz[N],cnt=,ans[N];
bool book[N];
void dfsfail(int u){
dfn[u]=++cnt,sz[u]=,book[u]=true;
for(int i=head[u];i;i=Next[i]){
int v=ver[i];
if(!book[v]) dfsfail(v),sz[u]+=sz[v];
}
}
void dfstrie(int u){
addd(dfn[u],);
if(End[u]){
for(int i=head1[End[u]];i;i=Next1[i]){
int v=ver1[i],x=dis[v];
// printf("%d %d %d\n",v,x,edge1[i]);
ans[edge1[i]]=sum(dfn[x]+sz[x]-)-sum(dfn[x]-);
}
}
for(int i=;i<;++i)
if(ch[u][i]) dfstrie(ch[u][i]);
addd(dfn[u],-);
}
char s[N];int len,m,st;
int main(){
// freopen("testdata.in","r",stdin);
scanf("%s",s+);
len=strlen(s+);
for(st=;st<=len;++st)
if(s[st]!='B'&&s[st]!='P') break;
int p=insert(,s[st]-'a');
for(int i=st+;i<=len;++i){
if(s[i]=='B') p=back(p);
else if(s[i]=='P') ed(p);
else p=insert(p,s[i]-'a');
}
scanf("%d",&m);
for(int i=,u,v;i<=m;++i){
scanf("%d%d",&u,&v);
add1(v,u,i);
}
build();
for(int i=;i<=tot;++i)
add(fail[i],i),add(i,fail[i]);
dfsfail(),dfstrie();
for(int i=;i<=m;++i) printf("%d\n",ans[i]);
return ;
}

洛谷P2414 [NOI2011]阿狸的打字机(AC自动机)的更多相关文章

  1. 洛谷 P2414 [NOI2011]阿狸的打字机 解题报告

    P2414 [NOI2011]阿狸的打字机 题目背景 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机. 题目描述 打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P'两个字母 ...

  2. 洛谷P2414 - [NOI2011]阿狸的打字机

    Portal Description 首先给出一个只包含小写字母和'B'.'P'的操作序列\(s_0(|s_0|\leq10^5)\).初始时我们有一个空串\(t\),依次按\(s_0\)的每一位进行 ...

  3. 【AC自动机】【树状数组】【dfs序】洛谷 P2414 [NOI2011]阿狸的打字机 题解

        这一题是对AC自动机的充分理解和树dfs序的巧妙运用. 题目背景 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机. 题目描述 打字机上只有28个按键,分别印有26个小写英文字母和' ...

  4. BZOJ 2434 Luogu P2414 [NOI2011]阿狸的打字机 (AC自动机、树状数组)

    题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=2434 题解: 我写的是离线做法,不知道有没有在线做法. 转化一波题意,\(x\)在AC ...

  5. P2414 [NOI2011]阿狸的打字机 AC自动机

    题意 给定n个模式串,有m个询问,每次询问第X个模式串在第Y个模中出现了多少次 解题思路 以fail树相反的方向建一棵树T,问题转化为X的子树中有多少个y的终止节点.跑出T的dfs序,X的子树就可以表 ...

  6. BZOJ 2434: [Noi2011]阿狸的打字机 [AC自动机 Fail树 树状数组 DFS序]

    2434: [Noi2011]阿狸的打字机 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 2545  Solved: 1419[Submit][Sta ...

  7. BZOJ 2434: [Noi2011]阿狸的打字机( AC自动机 + DFS序 + 树状数组 )

    一个串a在b中出现, 那么a是b的某些前缀的后缀, 所以搞出AC自动机, 按fail反向建树, 然后查询(x, y)就是y的子树中有多少是x的前缀. 离线, 对AC自动机DFS一遍, 用dfs序+树状 ...

  8. 【BZOJ2434】[NOI2011]阿狸的打字机 AC自动机+DFS序+树状数组

    [BZOJ2434][NOI2011]阿狸的打字机 Description 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机.打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P ...

  9. [NOI2011]阿狸的打字机 --- AC自动机 + 树状数组

    [NOI2011] 阿狸的打字机 题目描述: 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机. 打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P'两个字母.经阿狸研究发现, ...

随机推荐

  1. 2016WWDC详解

    今年苹果WWDC 2016上把所有系统都更新了个遍,watchOS.tvOS.macOS 和 iOS 都或多或少带来了新功能. 本文的主角是更新最多的 iOS 10,第一时间在一部 iPhone 6s ...

  2. HDU3247 Resource Archiver —— AC自动机 + BFS最短路 + 状压DP

    题目链接:https://vjudge.net/problem/HDU-3247 Resource Archiver Time Limit: 20000/10000 MS (Java/Others)  ...

  3. webpack为什么加载不了css?

    原文地址: https://segmentfault.com/q/1010000005099261 这个app是用react写的. webpack的loader设置是这样的 module:{ load ...

  4. jquery.dataTables.min.js: Uncaught TypeError: Cannot read property 'style' of undefined

    原因:datatable表格内容有操作列,而表头没有定义操作列 少写了一行:<th>操作</th>

  5. mooc_java 集合框架中 学生所选课程2Map&HashMap

    Map&HashMapMap提供映射关系,元素以键值对形式存储,Map的键值对一Entry类型的对象实例形式存在,key值不能重复,value可以键最多能映射到一个值,支持泛型 Map< ...

  6. 我所理解的RESTful Web API [设计篇]【转】

    原文:http://www.cnblogs.com/artech/p/restful-web-api-02.html <我所理解的RESTful Web API [Web标准篇]>Web服 ...

  7. 分享知识-快乐自己:初始 Struts2 (基本概念)及 搭建第一个Demo

    1):struts2 的基本概念: 1-1):Struts2 是什么? 1.Struts2是一个基于MVC设计模式的Web应用框架,它本质上相当于一个servlet,在MVC设计模式中,Struts2 ...

  8. Linux_配置_02_配置dns

    二.参考资料 1.centOS 7 设置DNS方法 同之前版本不同

  9. 【Python】numpy 数组拼接、分割

    摘自https://docs.scipy.org 1.The Basics 1.1 numpy 数组基础 NumPy’s array class is called ndarray. ndarray. ...

  10. Node初学者入门,一本全面的NodeJS教程

    作者: Manuel Kiessling  翻译: goddyzhao & GrayZhang & MondayChen 关于 本书致力于教会你如何用Node.js来开发应用,过程中会 ...