题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=2434

题目中这种多个串匹配的问题,一下子就想到了AC自动机。然后发现如果要建立AC自动机,跟着题目中的方式,不用把每个串提出来。如果是一个普通字符就直接加进去,如果是P就把当前节点记录下来,代表一个串,如果是B就相当于退回到trie树中的父亲节点。

建好AC自动机后来观察一下题目中的问题。这个询问相当于统计从根节点到代表y串的那个节点上的路径上,有多少个节点可以通过跳fail指针的方式到达x串的节点。

暴力统计显然是不行的,观察到fail路径上的每一个点的出度都为1,那么将fail全部反过来就是一棵树。问题就变成了在x的子树中,有多少个从根节点到y串的节点的路径上的节点。把fail树的dfs序求出来,我们可以用树状数组求子树和。

考虑离线。那么重新模拟题目中打字的过程,如果加入了一个字符,对应树状数组中此节点dfs序的位置+1,如果被删除了,则-1。这样就能实时维护树状数组中的点全部是y串中的点,于是提前处理一下,如果有关于y串的询问,答案就是x串的子树和,也就是dfs序对应的区间和。

 #include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
int inline readint(){
int Num;char ch;
while((ch=getchar())<''||ch>'');Num=ch-'';
while((ch=getchar())>=''&&ch<='') Num=Num*+ch-'';
return Num;
}
void outint(int x){
if(x>=) outint(x/);
putchar(x%+'');
}
char s[];
int Len,M;
int ch[][],fa[],sz=;
int pos[],tot=;
void Build_trie(){
int now=;
for(int i=;i<=Len;i++){
if(s[i]>='a'&&s[i]<='z'){
int idx=s[i]-'a';
if(!ch[now][idx]) ch[now][idx]=++sz;
fa[ch[now][idx]]=now;
now=ch[now][idx];
}
else if(s[i]=='B') now=fa[now];
else pos[++tot]=now;
}
}
int to[],ne[],fir[],cnt=;
void Addedge(int a,int b){
to[++cnt]=b;
ne[cnt]=fir[a];
fir[a]=cnt;
}
int fail[],q[];
void Set_fail(){
memset(fir,-,sizeof(fir));
int head=,tail=;
q[]=;
while(head<=tail){
int now=q[head];
for(int i=;i<;i++){
if(ch[now][i]){
q[++tail]=ch[now][i];
int tmp=now?ch[fail[now]][i]:;
fail[ch[now][i]]=tmp;
Addedge(tmp,ch[now][i]);
}
else ch[now][i]=ch[fail[now]][i];
}
head++;
}
}
int dfn=,in[],out[];
void Dfs(int x){
in[x]=++dfn;
for(int i=fir[x];i!=-;i=ne[i]) Dfs(to[i]);
out[x]=dfn;
}
struct Query{
int x,y,idx;
bool operator < (const Query &_)const{
return y<_.y;
}
}qry[];
int tree[];
int inline lowbit(int &x){
return x&-x;
}
void Add(int x,int d){
while(x<=dfn){
tree[x]+=d;
x+=lowbit(x);
}
}
int Sum(int x){
int sum=;
while(x){
sum+=tree[x];
x-=lowbit(x);
}
return sum;
}
int Ans[];
int main(){
scanf("%s",s+);
Len=strlen(s+);
Build_trie();
Set_fail();
Dfs();
M=readint();
for(int i=;i<=M;i++){
qry[i].x=readint();
qry[i].y=readint();
qry[i].idx=i;
}
sort(qry+,qry++M);
int now=,sk=,qk=;
for(int i=;i<=Len;i++){
if(s[i]>='a'&&s[i]<='z'){
now=ch[now][s[i]-'a'];
Add(in[now],);
}
else if(s[i]=='B'){
Add(in[now],-);
now=fa[now];
}
else{
sk++;
while(qry[qk].y==sk){
Ans[qry[qk].idx]=Sum(out[pos[qry[qk].x]])-Sum(in[pos[qry[qk].x]]-);
qk++;
}
}
}
for(int i=;i<=M;i++){
outint(Ans[i]);
putchar('\n');
}
return ;
}

[BZOJ2434][Noi2011]阿狸的打字机 AC自动机+树状数组+离线的更多相关文章

  1. BZOJ2434: [Noi2011]阿狸的打字机(AC自动机 树状数组)

    Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 4140  Solved: 2276[Submit][Status][Discuss] Descript ...

  2. 【BZOJ】2434: [Noi2011]阿狸的打字机 AC自动机+树状数组+DFS序

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

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

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

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

    题目链接 首先不需要存储每个字符串,可以将所有输入的字符依次存进Trie树,对于每个'P',记录该串结束的位置在哪,以及当前节点对应的是第几个串(当前串即根节点到当前节点):对于'B',只需向上跳一个 ...

  5. BZOJ2434[Noi2011]阿狸的打字机——AC自动机+dfs序+树状数组

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

  6. BZOJ2434:[NOI2011]阿狸的打字机(AC自动机,线段树)

    Description 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机.打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P'两个字母. 经阿狸研究发现,这个打字机是这样工作的 ...

  7. bzoj 2434 阿狸的打字机 - Aho-Corasick自动机 - 树状数组

    题目传送门 传送站I 传送站II 题目大意 阿狸有一个打字机,它有3种键: 向缓冲区追加小写字母 P:打印当前缓冲区(缓冲区不变) B:删除缓冲区中最后一个字符 然后多次询问第$x$个被打印出来的串在 ...

  8. 洛谷P2414 阿狸的打字机 [NOI2011] AC自动机+树状数组/线段树

    正解:AC自动机+树状数组/线段树 解题报告: 传送门! 这道题,首先想到暴力思路还是不难的,首先看到y有那么多个,菜鸡如我还不怎么会可持久化之类的,那就直接排个序什么的然后按顺序做就好,这样听说有7 ...

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

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

随机推荐

  1. table内 获取同一行 其他列的value

    table内  获取同一行 其他列的value function move(obj,ud){ var code = document.getElementById("reportName&q ...

  2. POJ3061 Subsequence

    Subsequence Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 16520   Accepted: 7008 Desc ...

  3. AOP原理

    AOP(面向切面编程):扩展功能时不修改源代码,采用横向抽取机制 纵向抽取机制: 横向抽取机制: 第一种情况: 第二种情况:

  4. POJ1163(基础线性DP)

    The Triangle Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 42547   Accepted: 25721 De ...

  5. OpenService 打开一个已经存在的服务

    SC_HANDLE WINAPI OpenService( _In_ SC_HANDLE hSCManager, _In_ LPCTSTR lpServiceName, _In_ DWORD dwDe ...

  6. SSE2 Intrinsics各函数介绍

    http://blog.csdn.net/fengbingchun/article/details/18460199

  7. g2o使用bug总结

    g2o进行3d2d优化的时候,设置优化图的边时,注意setVertex()中顶点的顺序. void setVertex(size_t i, Vertex* v) { assert(i < _ve ...

  8. Whitelabel Error Page异常

    做了个简单的springboot项目,一开始所有运行正常,之后写了一天的代码,再运行相同的代码,就一直报以下异常: Whitelabel Error Page This application has ...

  9. UVa 11520 Fill the Square (水题,暴力)

    题意:给n*n的格子里填上A-Z的字符,保证相邻字符不同,并且字典序最小. 析:直接从第一个格子开始暴力即可,每次判断上下左是不是相同即可. 代码如下: #pragma comment(linker, ...

  10. JavaWeb学习——获取类路径下的资源

    对于JavaWeb而言,获取类路径下的资源,就是获取classes目录下的资源. 获取资源的方式有两种,利用Class或ClassLoader. Class类的getResourceAsStream( ...