2434: [Noi2011]阿狸的打字机
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]阿狸的打字机的更多相关文章
- BZOJ 2434: [Noi2011]阿狸的打字机 [AC自动机 Fail树 树状数组 DFS序]
2434: [Noi2011]阿狸的打字机 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 2545 Solved: 1419[Submit][Sta ...
- BZOJ 2434: [Noi2011]阿狸的打字机( AC自动机 + DFS序 + 树状数组 )
一个串a在b中出现, 那么a是b的某些前缀的后缀, 所以搞出AC自动机, 按fail反向建树, 然后查询(x, y)就是y的子树中有多少是x的前缀. 离线, 对AC自动机DFS一遍, 用dfs序+树状 ...
- bzoj 2434 [Noi2011]阿狸的打字机 AC自动机
[Noi2011]阿狸的打字机 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 4001 Solved: 2198[Submit][Status][D ...
- 2434: [Noi2011]阿狸的打字机 - BZOJ
Description 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机.打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P'两个字母. 经阿狸研究发现,这个打字机是这样工作的 ...
- 【BZOJ】2434: [Noi2011]阿狸的打字机 AC自动机+树状数组+DFS序
[题意]阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机.打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P'两个字母. 经阿狸研究发现,这个打字机是这样工作的: l 输入小写 ...
- 【刷题】BZOJ 2434 [Noi2011]阿狸的打字机
Description 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机.打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P'两个字母. 经阿狸研究发现,这个打字机是这样工作的 ...
- BZOJ 2434: [Noi2011]阿狸的打字机 AC自动机+fail树+线段树
Description 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机.打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P'两个字母. 经阿狸研究发现,这个打字机是这样工作的 ...
- bzoj 2434 [Noi2011]阿狸的打字机(fail树+离线处理+BIT)
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2434 [题意] 按照一定规则生成n个字符串,回答若干个询问:(x,y),问第x个字符串 ...
- BZOJ 2434 [Noi2011]阿狸的打字机(AC自动机)
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2434 [题目大意] 给出一个打印的过程,'a'-'z'表示输入字母,P表示打印该字符串 ...
- bzoj 2434 [Noi2011]阿狸的打字机——AC自动机
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2434 dfs AC自动机,走过的点权值+1,回溯的时候权值-1:走到询问的 y 串的节点,看 ...
随机推荐
- ES6学习笔记(九)
1.概述 ES5的对象属性名都是字符串,这容易造成属性名的冲突.比如,你使用了一个他人提供的对象,但又想为这个对象添加新的方法(mixin模式),新方法的名字就有可能与现有方法产生冲突.如果有一种机制 ...
- PL/SQL Developer编码格式设置及中文乱码解决方案
1.PL/SQL Developer中文字段显示乱码 原因:因为数据库的编号格式和pl /sql developer的编码格式不统一造成的. 2.PL/SQL Developer编码格式设置详细的解决 ...
- 《C和指针》 读书笔记 -- 第13章 高级指针话题
1.函数指针 int (*f)(); int *(*f[])(); 用途: [1]回调函数 e.g. /*在一个单链表中查找指定值*/ Node *search_list(Node *node,voi ...
- 编码错误设置错误报 "SyntaxError: Non-ASCII character '/xe6' "
无意中碰到键盘导致一段处理中文拼音的 python 代码跑起来报了个错 “SyntaxError: Non-ASCII character ‘/xe6' " 看了下是注释 # coding: ...
- Beaglebone Back学习四(GPIO实验)
GPIO Beaglebone Back开发板引出了92个引脚,其中只有65个GPIO口可通过配置使用,由于引脚具有“复用”的特性,大约每个引脚有8种工作模式(Beagle System Refere ...
- IOS 学习参考
IOS 开发 http://code4app.com/ios/%E5%AE%9E%E6%97%B6%E6%9B%B4%E6%96%B0%E7%9A%84%E6%9B%B2%E7%BA%BF%E5%9B ...
- python学习笔记29(python中堆的使用)
堆(heap):优先队列的一种,使用优先队列能够以任意顺序增加对象,并且能在任意时间(可能在增加对象的同时)找到(也可能是移除)最小元素,比用于列表中min的方法要高效. Python中并没有独立的堆 ...
- Mac环境下装node.js,npm,express;(包括express command not found)
1. 下载node.js for Mac 地址: http://nodejs.org/download/ 直接下载 pkg的,双击安装,一路点next,很容易就搞定了. 安装完会提醒注意 node和n ...
- iOS开发(1) WebView和HTML 显示
iOS 7 已经release了.现在学习iOS开发还是非常热门的.到处也有些团队在寻找iOS开发的人才. 那么,iOS开发.....省略了1万字.... HTML5 +CSS3+JS...再省略1万 ...
- IntPtr
一:什么是IntPtr 先来看看MSDN上说的:用于表示指针或句柄的平台特定类型.这个其实说出了这样两个事实,IntPtr 可以用来表示指针或句柄.它是一个平台特定类型.对于它的解释,这个哥们写的比较 ...