BZOJ2434: [NOI2011]阿狸的打字机(AC自动机+dfs序+树状数组)
[NOI2011]阿狸的打字机
题目链接:https://www.luogu.org/problemnew/show/P2414
题目背景
阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机。
题目描述
打字机上只有28个按键,分别印有26个小写英文字母和'B'、'P'两个字母。经阿狸研究发现,这个打字机是这样工作的:
·输入小写字母,打字机的一个凹槽中会加入这个字母(这个字母加在凹槽的最后)。
·按一下印有'B'的按键,打字机凹槽中最后一个字母会消失。
·按一下印有'P'的按键,打字机会在纸上打印出凹槽中现有的所有字母并换行,但凹槽中的字母不会消失。
例如,阿狸输入aPaPBbP,纸上被打印的字符如下:
a aa ab 我们把纸上打印出来的字符串从1开始顺序编号,一直到n。打字机有一个非常有趣的功能,在打字机中暗藏一个带数字的小键盘,在小键盘上输入两个数(x,y)(其中1≤x,y≤n),打字机会显示第x个打印的字符串在第y个打印的字符串中出现了多少次。
阿狸发现了这个功能以后很兴奋,他想写个程序完成同样的功能,你能帮助他么?
输入输出格式
输入格式:
输入的第一行包含一个字符串,按阿狸的输入顺序给出所有阿狸输入的字符。
第二行包含一个整数m,表示询问个数。
接下来m行描述所有由小键盘输入的询问。其中第i行包含两个整数x, y,表示第i个询问为(x, y)。
输出格式:
输出m行,其中第i行包含一个整数,表示第i个询问的答案。
输入输出样例
aPaPBbP
3
1 2
1 3
2 3
2
1
0
说明
数据范围:
对于100%的数据,n<=100000,m<=100000,第一行总长度<=100000。

题解:
第一次做这种比较综合的题吧。。。
这个题还是要巧妙结合AC自动机中fail树的性质的,我们知道在fail树中,如果一个父亲结点指向一个儿子结点,那么以这个父亲结点为结尾的串包含在以儿子结点结尾的串中。这个题要求包含了多少次,其实就可以联想到构造一个fail树出来了。对于每个询问x,y,那么我们只需要统计在fail树中,以x为根节点的那颗子树上面包含了多少在1~y这条链上的结点就是了。
具体做法为离线做法,就是先对所有的询问x,y,连y->x的一条边。之后我们在fail树中跑一次dfs序,毕竟我们需要子树的信息嘛。
然后就从0开始,一个一个字符串加进去,如果出现小写字母,那么就让这个点的值加上1;如果出现“P”,就进行相应的回答;否则出现B,就让对应的值减去1。这里的操作可以在
回答询问的时候统计子树权值和就行了,这个过程可以用树状数组来实现。这些操作都可以在前缀树上面完成(实际上是一条链)。
代码如下:
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 1e5+;
char s[N];
int pos[N],head[N],fa[N],in[N],out[N],ans[N];
int tot,num;
queue <int> q;
struct Edge{
int v,next,id;
}e[N<<];
void adde(int u,int v,int id){
e[tot].v=v;e[tot].next=head[u];e[tot].id=id;head[u]=tot++;
}
struct Aho_Corasick{
int Size;
int ch[N][];
int val[N];
int fail[N];
void init(){
Size=-;
newnode();
}
int newnode(){
memset(ch[++Size],,sizeof(ch[]));
fail[Size]=;
return Size;
}
void insert(char *s){
int l=strlen(s);
int u=;
for(int i=;i<l;i++){
if(s[i]=='P'){
pos[++num]=u;
}else if(s[i]=='B'){
u=fa[u];
}else{
int idx=s[i]-'a';
if(!ch[u][idx]) ch[u][idx]=newnode();
fa[ch[u][idx]]=u;u=ch[u][idx];
}
}
}
void Getfail(){
while(!q.empty()) q.pop();
for(int i=;i<;i++){
if(ch[][i]) q.push(ch[][i]),adde(,ch[][i],);
}
while(!q.empty()){
int cur=q.front();q.pop();
for(int i=;i<;i++){
if(ch[cur][i]){
fail[ch[cur][i]]=ch[fail[cur]][i];
q.push(ch[cur][i]);
adde(ch[fail[cur]][i],ch[cur][i],);
}else{
ch[cur][i]=ch[fail[cur]][i];
}
}
}
}
}ac;
int dfn;
void dfs(int u,int pa){
in[u]=++dfn;
for(int i=head[u];i!=-;i=e[i].next){
int v=e[i].v;
if(v!=pa) dfs(v,u);
}
out[u]=dfn;
}
int c[N];
int lowbit(int x){
return x&(-x);
}
void add(int pos,int val){
for(int i=pos;i<=N-;i+=lowbit(i)) c[i]+=val;
}
int query(int l){
int ans=;
for(int i=l;i>;i-=lowbit(i)) ans+=c[i];
return ans ;
}
int main(){
scanf("%s",s);
memset(head,-,sizeof(head));tot=;
ac.init();
ac.insert(s);
ac.Getfail();
dfs(,-);
memset(head,-,sizeof(head));tot=;
int q;cin>>q;
for(int i=,x,y;i<=q;i++){
scanf("%d%d",&x,&y);
adde(y,x,i);
}
int l=strlen(s);
int cnt=,u=;;
for(int i=;i<l;i++){
if(s[i]=='B'){
add(in[u],-);u=fa[u];
}else if(s[i]=='P'){
cnt++;
int now=pos[cnt];
for(int j=head[cnt];j!=-;j=e[j].next){
int x=pos[e[j].v];
ans[e[j].id]=query(out[x])-query(in[x]-);
}
}else{
int idx=s[i]-'a';
u=ac.ch[u][idx];
add(in[u],);
}
}
for(int i=;i<=q;i++) printf("%d\n",ans[i]);
return ;
}
BZOJ2434: [NOI2011]阿狸的打字机(AC自动机+dfs序+树状数组)的更多相关文章
- BZOJ2434[Noi2011]阿狸的打字机——AC自动机+dfs序+树状数组
题目描述 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机.打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P'两个字母. 经阿狸研究发现,这个打字机是这样工作的: l 输入小 ...
- BZOJ 2434: [Noi2011]阿狸的打字机( AC自动机 + DFS序 + 树状数组 )
一个串a在b中出现, 那么a是b的某些前缀的后缀, 所以搞出AC自动机, 按fail反向建树, 然后查询(x, y)就是y的子树中有多少是x的前缀. 离线, 对AC自动机DFS一遍, 用dfs序+树状 ...
- 【BZOJ2434】[NOI2011]阿狸的打字机 AC自动机+DFS序+树状数组
[BZOJ2434][NOI2011]阿狸的打字机 Description 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机.打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P ...
- NOI 2011 阿狸的打字机 (AC自动机+dfs序+树状数组)
题目大意:略(太长了不好描述) 良心LOJ传送门 先对所有被打印的字符串建一颗Trie树 观察数据范围,并不能每次打印都从头到尾暴力建树,而是每遍历到一个字符就在Trie上插入这个字符,然后记录每次打 ...
- BZOJ 2434 阿狸的打字机(ac自动机+dfs序+树状数组)
题意 给你一些串,还有一些询问 问你第x个串在第y个串中出现了多少次 思路 对这些串建ac自动机 根据fail树的性质:若x节点是trie中root到t任意一个节点的fail树的祖先,那么x一定是y的 ...
- BZOJ_2434_[NOI2011]_阿狸的打字机_(AC自动机+dfs序+树状数组)
描述 http://www.lydsy.com/JudgeOnline/problem.php?id=2434 给出\(n\)个字符串,\(m\)个询问,对于第\(i\)个询问,求第\(x_i\)个字 ...
- CodeForces -163E :e-Government (AC自动机+DFS序+树状数组)
The best programmers of Embezzland compete to develop a part of the project called "e-Governmen ...
- BZOJ_3881_[Coci2015]Divljak_AC自动机+dfs序+树状数组
BZOJ_3881_[Coci2015]Divljak_AC自动机+dfs序+树状数组 Description Alice有n个字符串S_1,S_2...S_n,Bob有一个字符串集合T,一开始集合是 ...
- BZOJ2434: [Noi2011]阿狸的打字机(AC自动机 树状数组)
Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 4140 Solved: 2276[Submit][Status][Discuss] Descript ...
随机推荐
- UVa 401 - Palindromes 解题报告 - C语言
1.题目大意 输入字符串,判断其是否为回文串或镜像串.其中,输入的字符串中不含0,且全为合法字符.以下为所有的合法字符及其镜像: 2.思路 (1)考虑使用常量数组而不是if或switch来实现对镜像的 ...
- 创新手机游戏《3L》开发点滴(3)——道具、物品、装备表设计 V2(最终版)
我们正在开发一款新手游,里面有道具,之前也写了一篇博文记录了下我们的设计思路,但是国庆到了,于是我有了时间继续纠结这个问题. 其实我主要是在到底是用mysql还是mongodb上纠结.这个复杂.痛苦. ...
- MySQL Proxy使用
使用MySQL将读写请求转接到主从Server. 一 安装MySQL Proxy MySQL Proxy的二进制版非常方便,下载解压缩后即用. 解压缩的目录为: $mysql-proxy_instal ...
- 头文件#ifndef #define #endif使用
想必很多人都看过“头文件中的 #ifndef #define #endif 防止该头文件被重复引用”.但是是否能理解“被重复引用”是什么意思?是不能在不同的两个文件中使用include来包含这个头文件 ...
- 最小生成树——prim
prim:逐“点”生成最小生成树 与Dijkstra不同的是:加入点到生成树中,不要考虑与源点的距离,而是考虑与生成树的距离 #include <iostream> #include &l ...
- cp的使用
一.形式 cp [options] source1 source2 source3 .... directory 参数意义: 参数 意义 -i 当目标文件已存在时,会询问是否覆盖 -p 连同文件的属性 ...
- 20145214 《Java程序设计》第6周学习总结
20145214 <Java程序设计>第6周学习总结 教材学习内容总结 串流设计 Java将输入/输出抽象化为串流,数据有来源及目的地,衔接两者的是串流对象. 输入串流代表对象为java. ...
- 常用算法Java实现之直接插入排序
直接插入排序是将未排序的数据插入至已排好序序列的合适位置. 具体流程如下: 1.首先比较数组的前两个数据,并排序: 2.比较第三个元素与前两个排好序的数据,并将第三个元素放入适当的位置: 3.比较第四 ...
- javaIO--文件操作类
文件操作类主要是使用File类的各种方法对文件和目录进行操作.包括文件名.文件长度.最后修改时间和是否只读等,提供获得当前文件的路径名.判断文件是否存在.创建.删除文件和目录等一系列的操作方法. 下面 ...
- Debian以及Ubuntu源设置
在使用Debian和Ubuntu时,经常为了软件源烦恼,最近发现了一个网页,可以根据国家来设置源的地址,效果还不错. Debian:http://debgen.simplylinux.ch/ Ubun ...