2434: [Noi2011]阿狸的打字机

Time Limit: 10 Sec  Memory Limit: 256 MB

Submit: 3610  Solved: 1960

[Submit][Status][Discuss]

Description

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



经阿狸研究发现,这个打字机是这样工作的:



l 输入小写字母,打字机的一个凹槽中会加入这个字母(这个字母加在凹槽的最后)。



l 按一下印有'B'的按键,打字机凹槽中最后一个字母会消失。



l 按一下印有'P'的按键,打字机会在纸上打印出凹槽中现有的所有字母并换行,但凹槽中的字母不会消失。



例如,阿狸输入aPaPBbP,纸上被打印的字符如下:



a



aa



ab



我们把纸上打印出来的字符串从1开始顺序编号,一直到n。打字机有一个非常有趣的功能,在打字机中暗藏一个带数字的小键盘,在小键盘上输入两个数(x,y)(其中1≤x,y≤n),打字机会显示第x个打印的字符串在第y个打印的字符串中出现了多少次。



阿狸发现了这个功能以后很兴奋,他想写个程序完成同样的功能,你能帮助他么?

Input

输入的第一行包含一个字符串,按阿狸的输入顺序给出所有阿狸输入的字符。



第二行包含一个整数m,表示询问个数。



接下来m行描述所有由小键盘输入的询问。其中第i行包含两个整数x, y,表示第i个询问为(x, y)。

Output

输出m行,其中第i行包含一个整数,表示第i个询问的答案。

Sample Input

aPaPBbP



3



1 2



1 3



2 3


Sample Output

2



1



0

HINT

1<=N<=10^5

1<=M<=10^5
输入总长<=10^5

写出这道题,用裸的AC自动机肯定T

由题目字符串构建的形式,我们很容易想到trie树,可以很快建好一个AC自动机

对于询问(x,y),我们只要找出根到y末节点有多少个节点的fail指向x末端就好了【因为fail指针指向最长等于后缀的字符串】

直接统计?那也会T

直接统计是拿y以上节点去统计x,是多对一,非常浪费时间,但如果我们转化为一对多呢?就是拿x去找有多少属于y 的节点的fail指针指向x

这就引进了fail树

fail数,实质上就是把AC自动机的原边去掉,用fail边建立成的树

这样的树有一个很优美的性质,就是所有fail指针间接或直接指向u节点的节点共同组成u的子树,我们对于每个询问(x,y),我们只需查找x的子树中有多少个节点属于y就好了

怎么做呢?

求一个dfs序,可以用树状数组维护节点值,我们将属于y的询问放一起,属于y的节点全部+1,这样子只需要统计x子树之和就好了

具体操作时我们可以重新走一遍构造trie的路程,将路过的节点对应的值+1,向上时-1,每遇到一个单词末尾就将它的所有询问算出来

完美解决【md调了我一个下午= =,我代码能力还是太弱了】

调试点:

①树状数组的边界弄错

②节点+1 与 节点向下的次序弄反

③根节点标号不统一

我**真是太弱了

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cctype>
#include<queue>
#include<algorithm>
#define LL long long int
#define REP(i,n) for (int i = 1; i <= (n); i++)
#define fo(i,x,y) for (int i = (x); i <= (y); i++)
#define Redge(u) for (int k = head[u]; k != -1; k = edge[k].next)
#define lbt(x) (x & -x)
using namespace std;
const int maxn = 100005,maxm = 100005,INF = 1000000000;
inline int read(){
int out = 0,flag = 1;char c = getchar();
while (c < 48 || c > 57) {if (c == '-') flag = -1; c = getchar();}
while (c >= 48 && c <= 57) {out = out * 10 + c - 48; c = getchar();}
return out * flag;
}
int m,head[maxn],qh[maxn],ch[maxn][26],siz = 0,E[maxn],pre[maxn],fail[maxn],cnt = 0;
int id[maxn],Siz[maxn],nedge = 0,nq = 0;
int A[maxn],ans[maxn];
char s[maxn];
struct EDGE{int to,next;}edge[maxn],q[maxn];
inline void build(int u,int v){edge[nedge] = (EDGE){v,head[u]}; head[u] = nedge++;}
inline void add(int u,int v){while (u <= cnt) {A[u] += v; u += lbt(u);}}
inline int Query(int u){int ans = 0; while (u > 0) {ans += A[u]; u -= lbt(u);} return ans;}
inline int sum(int l,int r){return Query(r) - Query(l - 1);}
void insert(){
int u = 0,T = 0,i = 0,id;
while (isalpha(s[i])){
if (s[i] == 'P') E[++T] = u;
else if (s[i] == 'B') u = pre[u];
else {
if (!ch[u][id = s[i] - 'a']) {ch[u][id] = ++siz; pre[siz] = u;}
u = ch[u][id];
}
i++;
}
}
void getf(){
queue<int> q; int u,v;
for (int i = 0; i < 26; i++) if (ch[0][i]) q.push(ch[0][i]);
while (!q.empty()){
u = q.front();
q.pop();
for (int i = 0; i < 26; i++){
v = ch[u][i];
if (!v) ch[u][i] = ch[fail[u]][i];
else fail[v] = ch[fail[u]][i],q.push(v);
}
}
}
void dfs(int u){
id[u] = ++cnt; Siz[u] = 1; int to;
Redge(u) {
dfs(to = edge[k].to);
Siz[u] += Siz[to];
}
}
void solve(){
int u = 0,i = 0,p = 0;
while (isalpha(s[i])){
if (s[i] == 'B') add(id[u],-1),u = pre[u];
else if (s[i] == 'P'){
p++;
for (int k = qh[p]; k != -1; k = q[k].next){
int v = E[q[k].to]; /*cout<<q[k].to<<' '<<p<<endl;
cout<<" "<<u<<' '<<v<<endl;
cout<<" "<<id[v]<<' '<<id[v] + Siz[v] - 1<<endl;
cout<<sum(id[v],id[v] + Siz[v] - 1)<<endl;*/
ans[k] = sum(id[v],id[v] + Siz[v] - 1);
}
}else u = ch[u][s[i] - 'a'],add(id[u],1);
i++;
}
}
void init(){
memset(head,-1,sizeof(head));
memset(qh,-1,sizeof(qh));
scanf("%s",s);
insert(); getf();
for (int i = 1; i <= siz; i++) build(fail[i],i);
dfs(0);
m = read(); int x,y;
REP(i,m){
x = read(); y = read();
q[nq] = (EDGE){x,qh[y]}; qh[y] = nq++;
}
}
int main()
{
init();
//for (int i = 0; i <= siz; i++) cout<<id[i]<<' ';cout<<endl;
solve();
for (int i = 0; i < m; i++) printf("%d\n",ans[i]);
return 0;
}

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

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

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

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

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

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

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

  4. 【BZOJ-2434】阿狸的打字机 AC自动机 + Fail树 + DFS序 + 树状数组

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

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

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

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

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

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

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

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

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

  9. BZOJ 2434: [Noi2011]阿狸的打字机 AC自动机+fail树+线段树

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

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

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

随机推荐

  1. Ubuntu Server 下将HTML页面转换为PNG图片

    零.前言 最近做一个网站,需要将网页转换为图片.由于服务器是Ubuntu Server,没有图形界面,所以实现的过程中遇到了很多问题.记录下来备用. 一.安装CutyCapt CutyCapt是一个可 ...

  2. libevent学习七(bufferevent)

    1. 每个bufferevent 都拥有类型为struct evbuffer的input buffer和out buffer,分别供数据读取和数据写入使用. 2.读取和写入数据是通过编写和设置对应的回 ...

  3. 解决jQuery不同版同时引用的冲突

    今天研发的同事在开发一个新jQuery插件时,遇到一个揪心的问题.平台以前使用的 jQuery版本是1.2.6,偶,天啊!这是古代的版本啊! 由于很多功能基于老版本,不能删除啊,同志们都懂的! 于是我 ...

  4. SVG Sprite 使用Symbol元素制作ICON

    介绍 SVG是一种全新的使用方式,应该说这才是未来的主流,也是平台目前推荐的用法.之前写过两篇关于CSS icon在页面显示的博客,后来了解到现在大多数前端团队和项目都在使用SVG Sprite这种方 ...

  5. mapReduce入门教程

    什么是MapReduce MapReduce是Google提出的一个软件架构,用于大规模数据集(大于1TB)的并行运算.概念"Map(映射)"和"Reduce(归纳)&q ...

  6. Python为什么会打印两个\

    在Python里面,如果\后面不是一个合法的转移字符,那么,Python会打印两个\,换句话说,Python将\也当成普通字符看待,而不是转义符的标志: >>>S = 'C:\py\ ...

  7. 用纯css改变下拉列表select框的默认样式(转)

    用纯css改变下拉列表select框的默认样式 分享到 分类 JS学习   关键字 前端   发布 kris  2015-04-01 注意 转载须保留原文链接,译文链接,作者译者等信息.     在这 ...

  8. Python 字符串与基本语句

    Python特点 python中没有变量的声明 语句结束后没有分号 严格要求缩进 支持很长很长的大数运算(直接在Idle中输入即可) 用"#"来注释 BIF:Bulit-in fu ...

  9. unity像素贪吃蛇

    [ 星 辰 · 别 礼 ] 设计过程: 首先,在之前玩坏控制台做的那个c#贪吃蛇之后,我以为做unity会很简单,但事实比较不如人意...拖了好几天.因为过程中遇到一些问题 蛇身的移动,还是用列表,将 ...

  10. android BadgeView的使用(图片上的文字提醒)

    BadgeView主要是继承了TextView,所以实际上就是一个TextView,底层放了一个label,可以自定义背景图,自定义背景颜色,是否显示,显示进入的动画效果以及显示的位置等等: 这是Gi ...