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

【题目大意】

  给出一个打印的过程,'a'-'z'表示输入字母,P表示打印该字符串
  B表示删去一个字符。问第x个打印的字符串在第y个打印的字符串中出现的次数

【题解】

  我们根据打印的过程建立trie树,
  当x是y的子串当且仅当y到根的链上有fail指针指向x的结尾,
  而x在y中的出现次数则取决于有几个这样的指针,
  我们根据fail指针建立fail树,按照fail树的dfs序进行统计,
  在每个y处记录其要查询的x,在y点用树状数组对x点求dfs序区间和即可。

【代码】

#include <cstdio>
#include <algorithm>
#include <vector>
#include <cstring>
using namespace std;
const int N=100010;
int dfn,l[N],r[N],ans[N];
vector<int> v[N],Q[N],ID[N];
namespace BIT{
int c[N<<1]; //dfs
void Initialize(){memset(c,0,sizeof(c));}
void add(int x,int v){while(x<=dfn)c[x]+=v,x+=x&-x;}
int query(int x){int res=0;while(x)res+=c[x],x-=x&-x;return res;}
}
namespace AC_DFA{
const int Csize=26;
int id,tot,son[N][Csize],sum[N],f[N],fail[N],q[N],pos[N],match[N];
void Initialize(){
memset(sum,0,sizeof(int)*(tot+1));
memset(fail,0,sizeof(int)*(tot+1));
for(int i=0;i<=tot;i++)for(int j=0;j<Csize;j++)son[i][j]=0;
tot=0; id=0; fail[0]=-1;
}
inline int Tr(char ch){return ch-'a';}
void Build(char *s){
int x=0;
for(int l=strlen(s),i=0,w;i<l;i++){
if(s[i]=='P')pos[++id]=x;
else if(s[i]=='B')x=f[x];
else{
if(!son[x][w=Tr(s[i])]){
son[x][w]=++tot;
f[tot]=x;
}x=son[x][w];
}
}
}
void MakeFail(){
int h=1,t=0,i,j,x=0;
for(i=0;i<Csize;i++)if(son[0][i])q[++t]=son[0][i];
while(h<=t)for(x=q[h++],i=0;i<Csize;i++)
if(son[x][i]){
fail[son[x][i]]=son[fail[x]][i],q[++t]=son[x][i];
}else son[x][i]=son[fail[x]][i];
}
void Solve(char *s){
using namespace BIT;
BIT::Initialize();
int x=0,id=0;
add(l[0],1);
for(int L=strlen(s),i=0;i<L;i++){
if(s[i]=='P'){
id++;
for(int k=0;k<Q[id].size();k++){
int u=pos[Q[id][k]];
ans[ID[id][k]]=query(r[u])-query(l[u]-1);
}
}else if(s[i]=='B')add(l[x],-1),x=f[x];
else x=son[x][Tr(s[i])],add(l[x],1);
}
}
}
void Dfs(int x){
l[x]=++dfn;
for(int i=0;i<v[x].size();i++)Dfs(v[x][i]);
r[x]=++dfn;
}
char s[N];
int main(){
using namespace AC_DFA;
Initialize();
scanf("%s",s);
Build(s); MakeFail();
for(int i=1;i<=tot;i++)v[fail[i]].push_back(i);
int m,x,y;
scanf("%d",&m);
for(int i=1;i<=m;i++){
scanf("%d%d",&x,&y);
Q[y].push_back(x);
ID[y].push_back(i);
}Dfs(0); Solve(s);
for(int i=1;i<=m;i++)printf("%d\n",ans[i]);
return 0;
}

BZOJ 2434 [Noi2011]阿狸的打字机(AC自动机)的更多相关文章

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

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

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

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

  3. bzoj 2434 [Noi2011]阿狸的打字机 AC自动机

    [Noi2011]阿狸的打字机 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 4001  Solved: 2198[Submit][Status][D ...

  4. bzoj 2434 [Noi2011]阿狸的打字机——AC自动机

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2434 dfs AC自动机,走过的点权值+1,回溯的时候权值-1:走到询问的 y 串的节点,看 ...

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

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

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

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

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

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

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

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

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

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

  10. 【刷题】BZOJ 2434 [Noi2011]阿狸的打字机

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

随机推荐

  1. 认识单点登录cas

    么是单点登录?单点登录全称Single Sign On(以下简称SSO),是指在多系统应用群中登录一个系统,便可在其他所有系统中得到授权而无需再次登录,包括单点登录与单点注销两部分 1.登录 相比于单 ...

  2. java规范之checkstyle

    1. 概述 随着中心的代码规范的建立和实施,项目组对代码规范要求,以及软件工程师们对自身代码的编写规范重要性的认知,“代码规范”已经成为了中心的一个“热词”.然后怎么才能写出有规范的代码,怎么才能养成 ...

  3. 关于linux下crontab mysql备份出来的数据为0字节的问题

    问题出在计划任务所执行的脚本上! 脚本中的调用的指令应该都写全路径~ 实例: # crontab -c 编辑下的内容 30 18 * * * /root/backup.sh 意思为:每天18:30执行 ...

  4. linux——vi和vim的区别

    vi 和vim 的区别   它们都是多模式编辑器,不同的是vim 是vi的升级版本,它不仅兼容vi的所有指令,而且还有一些新的特性在里面. vim的这些优势主要体现在以下几个方面:1.多级撤消我们知道 ...

  5. 树莓派开发系列教程3--ssh、vnc远程访问

    注意:树莓派系列的3篇文章里面的图片因为博客转移过程丢失了,非常抱歉 前言 远程访问有很多种方式可以实现.比如ssh.telnet.ftp.samba.远程桌面等等,各有优缺点.本文主要以ssh和远程 ...

  6. 树莓派开发系列教程2--树莓派3B介绍及系统烧写、基本配置

    注意:树莓派系列的3篇文章里面的图片因为博客转移过程丢失了,非常抱歉 前言 树莓派系列里涉及到需要额外电脑来协助处理的部分(如烧写系统,远程访问等等),统一在运行着Ubuntu15系统的mac air ...

  7. python基础===string模块常量

    In [8]: import string In [9]: dir(string) In [10]: string.ascii_letters Out[10]: 'abcdefghijklmnopqr ...

  8. [MySQL] AUTO_INCREMENT lock Handing in InnoDB

    MySQL AUTO_INCREMENT lock Handing in InnoDB 在MySQL的表设计中很普遍的使用自增长字段作为表主键, 实际生产中我们也是这样约束业务开发同学的, 其中的优势 ...

  9. ThoughtWorks代码挑战——FizzBuzzWhizz游戏 通用高速版(C/C++ & C#)

    最早看到这个题目是从@ 程序媛想事儿(Alexia) 的 最难面试的IT公司之ThoughtWorks代码挑战——FizzBuzzWhizz游戏 开始的,然后这几天陆陆续续有N个小伙伴发表了自己的文章 ...

  10. PHP 利用nginx的X-sendfile控制下载,提高下载效率

    https://blog.csdn.net/qq_34839657/article/details/52812885 https://www.jianshu.com/p/bf5c387830b7 为了 ...