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

题解:

我写的是离线做法,不知道有没有在线做法。

转化一波题意,\(x\)在AC自动机上代表的字符串在\(y\)代表的字符串中出现的次数等于\(x\)在fail树的子树内有多少个点是\(y\)点Trie树上的祖先。

然后不难得到做法: 将询问离线按照\(y\)在AC自动机上的id排序(一个正常的构建Trie的方式建出来应该就满足节点的编号是它的Trie树DFS序).

在Trie树上DFS, 同时回答询问, 用树状数组维护即可

代码

#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
using namespace std; const int N = 1e5;
const int S = 26;
struct Edge
{
int v,nxt;
} e[(N<<1)+3];
struct Query
{
int x,y,id;
bool operator <(const Query &arg) const
{
return y<arg.y;
}
} qr[N+3];
int fe[N+3];
int dfn[N+3];
int fa[N+3];
int son[N+3][S+3];
int fail[N+3];
char str[N+3];
int stk[N+3];
int id[N+3];
int que[N+3];
int bit[N+3];
int ord[N+3];
int ans[N+3];
int sz[N+3];
int n,q,siz,m,cnt,en; void addval(int lrb,int val)
{
while(lrb<=cnt)
{
bit[lrb] += val;
lrb += (lrb&(-lrb));
}
} int querysum(int rb)
{
int ret = 0;
while(rb)
{
ret += bit[rb];
rb -= (rb&(-rb));
}
return ret;
} void addedge(int u,int v)
{
en++; e[en].v = v;
e[en].nxt = fe[u]; fe[u] = en;
} void buildTrie()
{
int u = 0,tp = 0; stk[tp] = 0;
for(int i=1; i<=n; i++)
{
if(str[i]=='B')
{
stk[tp] = 0; tp--;
u = stk[tp];
}
else if(str[i]=='P')
{
m++; id[m] = u;
}
else
{
str[i]-=96;
if(!son[u][str[i]]) {siz++; son[u][str[i]] = siz;}
u = son[u][str[i]];
tp++; stk[tp] = u;
}
}
} void buildACA()
{
int head = 1,tail = 0;
for(int i=1; i<=S; i++) {if(son[0][i]) {tail++; que[tail] = son[0][i];} fail[son[0][i]] = 0;}
while(head<=tail)
{
int u = que[head]; head++;
addedge(fail[u],u); addedge(u,fail[u]);
for(int i=1; i<=S; i++)
{
if(son[u][i]) {fail[son[u][i]] = son[fail[u]][i]; tail++; que[tail] = son[u][i];}
else {son[u][i] = son[fail[u]][i];}
}
}
} void dfs(int u)
{
cnt++; dfn[u] = cnt;
sz[u] = 1;
for(int i=fe[u]; i; i=e[i].nxt)
{
if(e[i].v==fa[u]) continue;
fa[e[i].v] = u;
dfs(e[i].v);
sz[u] += sz[e[i].v];
}
} int main()
{
scanf("%s",str+1); n = strlen(str+1);
scanf("%d",&q);
buildTrie();
buildACA();
for(int i=1; i<=q; i++)
{
int x,y; scanf("%d%d",&x,&y);
qr[i].x = id[x]; qr[i].y = id[y]; qr[i].id = i;
}
dfs(0);
sort(qr+1,qr+q+1);
int u = 0,tp = 0,j = 0; stk[tp] = 0; addval(dfn[u],0);
for(int i=1; i<=n; i++)
{
if(str[i]=='B')
{
addval(dfn[u],-1);
stk[tp] = 0; tp--;
u = stk[tp];
}
else if(str[i]=='P')
{
while(j<q && qr[j+1].y==u)
{
j++;
ans[qr[j].id] = querysum(dfn[qr[j].x]+sz[qr[j].x]-1)-querysum(dfn[qr[j].x]-1);
}
}
else
{
u = son[u][str[i]];
tp++; stk[tp] = u;
addval(dfn[u],1);
}
}
for(int i=1; i<=q; i++)
{
printf("%d\n",ans[i]);
}
return 0;
}

BZOJ 2434 Luogu P2414 [NOI2011]阿狸的打字机 (AC自动机、树状数组)的更多相关文章

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

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

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

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

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

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

  4. [BZOJ2434][Noi2011]阿狸的打字机 AC自动机+树状数组+离线

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=2434 题目中这种多个串匹配的问题,一下子就想到了AC自动机.然后发现如果要建立AC自动机, ...

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

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

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

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

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

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

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

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

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

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

随机推荐

  1. js添加页面元素

    js动态创建html元素需要使用到下面这些常见的js方法. getElementById();//返回带有指定 ID 的元素. getElementsByTagName();//返回包含带有指定标签名 ...

  2. 4.4系统,拍照-裁剪,resultCode返回0

    问题描述: take photo -> 拍照 -> 确定 -> 截图 -> 保存,此时返回给onActivityResult的resultCode是0,截图无效.我查看图片储存 ...

  3. UNDO表空间不足解决方法

    确认UNDO表空间名称 select name from v$tablespace; 检查数据库UNDO表空间占用空间情况以及数据文件存放位置: select file_name,bytes/1024 ...

  4. codevs4511信息传递(Tarjan求环)

    题目描述 有n个同学(编号为1到n)正在玩一个信息传递的游戏.在游戏里每人都有一个固定的信息传递对象,其中,编号为i的同学的信息传递对象是编号为Ti同学. 游戏开始时,每人都只知道自己的生日.之后每一 ...

  5. 题解报告:hdu 2149 Public Sale(巴什博弈)

    题目链接:acm.hdu.edu.cn/showproblem.php?pid=2149 Problem Description 虽然不想,但是现实总归是现实,Lele始终没有逃过退学的命运,因为他没 ...

  6. ios cocos2d-x 多点触摸

    需要在 AppController.mm文件中添加一句多点触摸支持: [_view setMultipleTouchEnabled:true];

  7. Serializable和Parcelable的简单介绍

    Serializable和Pacelable接口可以完成对象的序列化过程,当我们需要通过Intent和Binder传输数据时就需要使用Parcelable或者Serializable. Seriali ...

  8. Raspberry Pi开发之旅-空气温湿度检测(DHT11)

    一.首先,简单介绍下DHT11: DHT11是一个温湿度传感器,分为3个接口,分别为:VCC, DATA, GND  引脚号 名称 类型 说明 1 VCC 电源 +级,输入3V-5.5V 2 DATA ...

  9. unity3d 各键值对应代码

    KeyCode :KeyCode是由Event.keyCode返回的.这些直接映射到键盘上的物理键.  值        对应键 Backspace     退格键 Delete      Delet ...

  10. Angular——tab切换案例

    基本介绍 angular框架下的tab切换,相比较于之前的纯js写的代码,有一个很大的特点就是以数据为驱动,基本上不用搜索dom元素就可以实现效果 基本使用 (1)导航部分使用的是的状态使用的是ng- ...