ac自动机,bit,dfs序。

本文所有的stl都是因为自己懒得实现。

 

首先x在y里面出现,就意味y节点可以顺着fail回去。

反向建出一个fail数,然后搞出dfs序列。找出x对应的区间有多少个y。

再用离线操作,把每个y需要计算的x事先保存下来。

#include<cstdio>
#include<algorithm>
#include<cstring>
#include<queue>
#include<vector>
using namespace std;
const int maxn = 100000 + 10; char s[maxn];
int a[maxn][26];
int g[maxn],v[maxn],next[maxn],eid;
int pos[maxn],cnt;
int fa[maxn],fail[maxn],L[maxn],R[maxn],query[maxn][3];
int m,n,u,dfn,vid;
vector<int> Q[maxn];
queue<int> q; struct BIT {
int a[maxn<<1],n; int lowbit(int x) {
return (x & -x);
} void add(int x,int d) {
//printf("c %d %d\n",x,d);
for(;x<=n;x+=lowbit(x)) a[x]+=d;
//printf("c");
} int query(int x) {
int res=0;
for(;x;x-=lowbit(x)) res+=a[x];
return res;
} void init(int _n) {
n=_n;
}
}bit; void addedge(int a,int b) {
v[eid]=b; next[eid]=g[a]; g[a]=eid++;
} void dfs(int u) {
L[u]=++dfn;
for(int i=g[u];~i;i=next[i]) dfs(v[i]);
R[u]=dfn;
} void get_trie() {
for(int i=0;i<26;i++) a[0][i]=1; int p=1,c;vid=1;
for(int i=0;i<n;i++) {
//printf("f[%d] = %d\n",i,p);
if(s[i]=='P') pos[++cnt]=p;
else if(s[i]=='B') p=fa[p];
else {
c=s[i]-'a';
if(!a[p][c])
a[p][c]=++vid,fa[vid]=p;
p=a[p][c];
}
//printf("f[%d] = %d\n",i,p);
}
} void debug(int p) {
} void get_fail() {
fail[1]=0;
q.push(1); while(!q.empty()) {
u=q.front();q.pop();
for(int k=0,p;k<26;k++)
if(a[u][k]) {
for(p=fail[u];p&&!a[p][k];p=fail[p]);
fail[a[u][k]]=a[p][k];
q.push(a[u][k]);
}
}
} void get_tree() {
for(int i=1;i<=vid;i++) addedge(fail[i],i);
dfs(1);
} void build() {
memset(g,-1,sizeof(g));
scanf("%s",s);n=strlen(s); get_trie();
//debug(1);
get_fail();
get_tree(); scanf("%d",&m);
for(int i=1;i<=m;i++) {
scanf("%d %d",&query[i][0],&query[i][1]);
query[i][0]=pos[query[i][0]];
query[i][1]=pos[query[i][1]];
Q[query[i][1]].push_back(i);
} } void solve() {
bit.init(n<<1);
int p=1;
for(int i=0;i<n;i++) {
//printf("s[i]=%c\n",s[i]);
if(s[i]=='P') for(int j=0;j<Q[p].size();j++)
query[Q[p][j]][2]=bit.query(R[query[Q[p][j]][0]])-bit.query(L[query[Q[p][j]][0]]-1);
else if(s[i]=='B') bit.add(L[p],-1),p=fa[p];
else p=a[p][s[i]-'a'],bit.add(L[p],1);
//printf(" %d\n",i);
} for(int i=1;i<=m;i++) printf("%d\n",query[i][2]);
} int main() {
build();
solve(); return 0;
}

2434: [Noi2011]阿狸的打字机的更多相关文章

  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. 2434: [Noi2011]阿狸的打字机 - BZOJ

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

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

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

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

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

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

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

  8. bzoj 2434 [Noi2011]阿狸的打字机(fail树+离线处理+BIT)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2434 [题意] 按照一定规则生成n个字符串,回答若干个询问:(x,y),问第x个字符串 ...

  9. BZOJ 2434 [Noi2011]阿狸的打字机(AC自动机)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2434 [题目大意] 给出一个打印的过程,'a'-'z'表示输入字母,P表示打印该字符串 ...

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

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

随机推荐

  1. ES6学习笔记(二)

    1.数组的解构赋值 基本用法 ES6允许按照一定模式,从数组和对象中提取值,对变量进行赋值,这被称为解构(Destructuring). 以前,为变量赋值,只能直接指定值. var a = 1; va ...

  2. ubuntu server版连接vpn服务器

    命令行的方法: 1.要下载pptp的客户端    sudo apt-get install pptp-linux 2.创建连接     sudo pptpsetup –create vpn001 –s ...

  3. WordPress非插件添加文章浏览次数统计功能

    一: 转载:http://www.jiangyangangblog.com/26.html 首先在寻找到functions.php.php文件夹,在最后面  ?> 的前面加入下面的代码 func ...

  4. Delphi Base64 编解码函数

    Delphi 自带 Base64 编解码的单元, EncdDecd这个单元提供两套四个公开函数: 对流的编解码:procedure EncodeStream(Input, Output: TStrea ...

  5. Spark Tungsten揭秘 Day2 Tungsten-sort Based Shuffle

    Spark Tungsten揭秘 Day2 Tungsten-sort Based Shuffle 今天在对钨丝计划思考的基础上,讲解下基于Tungsten的shuffle. 首先解释下概念,Tung ...

  6. 如何将无线路由器作为交换机,将光猫(路由器A)分出来的一条网线接到自家另一台路由器B上,最大化利用网络资源

    从隔壁邻居只接了一条网线过来,由于无线网络的距离有限,不能覆盖到家里任何角落,然而,我又想家里一台台式电脑和无线设备都能够连接wifi进行上网。 摸索了一个上午,知道将家里的无线路由器B当作一个无线A ...

  7. 【分享】SQL Server优化50法

    虽然查询速度慢的原因很多,但是如果通过一定的优化,也可以使查询问题得到一定程度的解决. 查询速度慢的原因很多,常见如下几种: 没有索引或者没有用到索引(这是查询慢最常见的问题,是程序设计的缺陷) I/ ...

  8. 类的本质、description方法、SEL、NSLog输出增强

    一.类的本质 1.类也是个对象 其实类也是一个对象,是Class类型的对象,简称“类对象” Class类型的定义 typedef struct objc_class *Class; 类名就代表着类对象 ...

  9. bnuoj 20838 Item-Based Recommendation (模拟)

    http://www.bnuoj.com/bnuoj/problem_show.php?pid=20838 [题意]: 有点长,略. [code]: #include <iostream> ...

  10. window.location.hash属性介绍

    location是javascript里边管理地址栏的内置对象,比如location.href就管理页面的url,用location.href=url就可以直接将页面重定向url.而location. ...