bzoj 2434 fail tree+dfs序
首先比较明显的是我们可以将字符串组建立ac自动机,那么对于询问s1字符串在s2字符串中出现的次数,就是在以s1结尾为根的fail tree中,子树有多少个节点是s2的节点,这样我们处理fail tree的dfs序,然后用BIT维护,但是如果只是在线处理询问的话会tle,因为每个询问需要将节点的每一位在BIT中都修改,那么我们就浪费了好多性质,因为对于好多字符串拥有较长的LCP,那么我们可以模拟建trie的过程在自动机上跑,每遇到一个添加的字符,就解决所有询问为某字符串在该字符串中出现的次数,这样就可以了。
/**************************************************************
Problem: 2434
User: BLADEVIL
Language: C++
Result: Accepted
Time:852 ms
Memory:39128 kb
****************************************************************/
//By BLADEVIL
#include <cstdio>
#include <cstring>
#define maxn 200010
using namespace std;
struct node{
int cnt,last,left,right;
node *child[],*fail,*father;
node(){
cnt=last=left=right=;
fail=father=NULL;
memset(child,,sizeof child);
}
}nodepool[maxn],*totnode,*root,*que[maxn],*adr[maxn],*other[maxn],*adrans[maxn];
char c[maxn];
int len,tot,l,sum;
int pre[maxn],bit[maxn];
int ll,preans[maxn],otherans[maxn],lastans[maxn],sizeans[maxn];
int ans[maxn];
void add(int x,int y){
while (x<=sum){
bit[x]+=y;
x+=x&(-x);
}
}
int ask(int x){
int ans=;
while (x){
ans+=bit[x];
x-=x&(-x);
}
return ans;
}
void connect(node *x,node *y){
pre[++l]=x->last;
x->last=l;
other[l]=y;
//printf("%d %d\n",x,y);
}
void connectans(int x,int y,int z){
preans[++l]=lastans[x];
lastans[x]=l;
otherans[l]=y;
sizeans[l]=z;
}
void build_trie(){
totnode=nodepool; root=totnode++;
scanf("%s",&c); len=strlen(c);
node *t=root;
for (int i=;i<len;i++){
if (c[i]=='P') adrans[++tot]=t; else
if (c[i]=='B') t=t->father; else {
if (!t->child[c[i]-'a'])
t->child[c[i]-'a']=totnode++,t->child[c[i]-'a']->father=t;
t=t->child[c[i]-'a'];
adr[i]=t;
};
//printf("%d ",t);
}
//printf("\n");
//for (int i=0;i<len;i++) printf("%d ",adr[i]);
//for (node *i=nodepool;i!=totnode;i++) printf("%d %d\n",i,i->father);
}
void build_ac(){
int h=,t=;
que[]=root; root->fail=root;
for (int i=;i<;i++) if (!root->child[i]) root->child[i]=root;
while (h<t){
node *v=que[++h];
for (int i=;i<;i++) if (v->child[i]&&v->child[i]!=root){
que[++t]=v->child[i];
node *u=v->fail;
que[t]->fail=u->child[i]!=que[t]?u->child[i]:root;
} else v->child[i]=v->fail->child[i];
}
//for (int i=1;i<=t;i++) printf("%d %d ",que[i],que[i]->fail); printf("\n");
//for (int i=1;i<=tot;i++) printf("%d ",adr[i]); printf("\n");
}
void dfs(node *x,node *fa){
//printf("%d %d %d\n",x,fa,sum);
x->left=++sum;
for (int p=x->last;p;p=pre[p]){
if (other[p]==fa) continue;
dfs(other[p],x);
}
x->right=sum;
}
/*void work(){
for (node *i=nodepool;i!=totnode;i++) if (i!=root) connect(i->fail,i);
dfs(root,NULL);
//for (node *i=nodepool;i!=totnode;i++) printf("%d %d %d %d\n",i,i->left,i->right,i->father);
int m;
scanf("%d",&m);
while (m--){
int x,y;
scanf("%d %d",&x,&y);
for (node *i=adr[y];i!=root;i=i->father) add(i->left,1);//printf("%d ",i->left);
//printf("%d %d",adr[x]->left,adr[x]->right);
//printf("%d ",ask(adr[x]->right));
printf("%d\n",ask(adr[x]->right)-ask(adr[x]->left-1));
for (node *i=adr[y];i!=root;i=i->father) add(i->left,-1);
}
}*/
void work(){
for (node *i=nodepool;i!=totnode;i++) if (i!=root) connect(i->fail,i);
dfs(root,NULL);
int m;
scanf("%d",&m);
for (int i=;i<=m;i++){
int x,y;
scanf("%d%d",&x,&y);
connectans(y,x,i);
}
int stack[maxn],tot=,ansy=;
memset(stack,,sizeof stack);
for (int i=;i<len;i++){
if (c[i]=='P'){
ansy++;
for (int p=lastans[ansy];p;p=preans[p]){
ans[sizeans[p]]=ask(adrans[otherans[p]]->right)-ask(adrans[otherans[p]]->left-);
}
} else
if (c[i]=='B') {
add(adr[stack[tot--]]->left,-);
} else {
stack[++tot]=i;
add(adr[i]->left,);
}
}
for (int i=;i<=m;i++) printf("%d\n",ans[i]);
}
int main(){
build_trie();
build_ac();
work();
return ;
}
bzoj 2434 fail tree+dfs序的更多相关文章
- POJ.3321 Apple Tree ( DFS序 线段树 单点更新 区间求和)
POJ.3321 Apple Tree ( DFS序 线段树 单点更新 区间求和) 题意分析 卡卡屋前有一株苹果树,每年秋天,树上长了许多苹果.卡卡很喜欢苹果.树上有N个节点,卡卡给他们编号1到N,根 ...
- BZOJ 2905: 背单词 AC自动机+fail树+dfs序+线段树
Description 给定一张包含N个单词的表,每个单词有个价值W.要求从中选出一个子序列使得其中的每个单词是后一个单词的子串,最大化子序列中W的和. Input 第一行一个整数TEST,表示数据组 ...
- bzoj2434 fail树 + dfs序 + 树状数组
https://www.lydsy.com/JudgeOnline/problem.php?id=2434 打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P'两个字母.经阿狸研究发现, ...
- 【Codeforces163E】e-Government AC自动机fail树 + DFS序 + 树状数组
E. e-Government time limit per test:1 second memory limit per test:256 megabytes input:standard inpu ...
- BZOJ2434 [Noi2011]阿狸的打字机(AC自动机 + fail树 + DFS序 + 线段树)
题目这么说的: 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机.打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P'两个字母.经阿狸研究发现,这个打字机是这样工作的: 输入小 ...
- NOI2011阿狸的打字机(fail树+DFS序)
Description 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机.打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P'两个字母. 经阿狸研究发现,这个打字机是这样工作的 ...
- poj 3321 Apple Tree dfs序+线段树
Apple Tree Time Limit: 2000MS Memory Limit: 65536K Description There is an apple tree outsid ...
- [poj3321]Apple Tree(dfs序+树状数组)
Apple Tree Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 26762 Accepted: 7947 Descr ...
- Codeforces Round #225 (Div. 1) C. Propagating tree dfs序+树状数组
C. Propagating tree Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/383/p ...
随机推荐
- 原生js移动端可拖动进度条插件
该插件最初的想法来自网上的一篇文章,直达链接:https://www.cnblogs.com/libin-1/p/6220056.html 笔者因为业务需要寻找到这个插件,然后拿来用之,发现各种不方便 ...
- tomcat web页面管理应用配置
大部分时候,我们的tomcat服务器都不是部署在本机,那么怎么样不通过ftp/sftp方式来将war包部署到tomcat容器呢? tomcat有提供web页面管理应用的功能. 我们来看看怎么配置实现该 ...
- c++ new 堆 栈
根据32位的Windows系统默认有2GB的用户空间,则不能new超过2GB的,执行下列代码: ***]; 会出现下面的错误 error C2148: 数组的总大小不得超过 0x7fffffff 字节 ...
- centos7编译安装redis遇坑
编译redis时:make cc Command not found 原因分析:没有安装gcc,执行: yum install gcc 编译redis时:error: jemalloc/jemallo ...
- P3074 [USACO13FEB]牛奶调度Milk Scheduling
题目描述 Farmer John's N cows (1 <= N <= 10,000) are conveniently numbered 1..N. Each cow i takes ...
- [luogu2783] 有机化学之神偶尔会做作弊
题目链接 洛谷. Solution 边双缩点然后\(lca\)跑\(dis\)就好了. 注意这里是边双,不知道为啥所有题解都说的是点双. 边双是定义在点上的,即每个点只属于一个边双:点双是定义在边上的 ...
- [HAOI2006]受欢迎的牛 tarjan缩点 + 拓扑排序
---题面--- 题解: 首先tarjan缩点应该还是容易想到的,因为喜爱具有传递性,所以一个强联通分量里面的点实际上是全部等效的,所以我们可以缩成一个方便判断, 缩完点之后整张图就变成了一个有向无环 ...
- redux的bindActionCreators
bindActionCreators是redux的一个API,作用是将单个或多个ActionCreator转化为dispatch(action)的函数集合形式. 开发者不用再手动dispatch(ac ...
- [NOI2017]蔬菜——时光倒流+贪心
题目链接 题解: 貌似一眼看过去是一个贪心. 其他的算法要记录的东西就太多了. 部分分其实很高.但是没有什么提示. 想一些套路:二分?不行还要贪心判断. 分治?前后取法是有影响的. 时光倒流? 也许可 ...
- 第三方库安装——lxml
环境 操作系统:CentOS 6.7 32-bit Python:2.6.6 安装 安装依赖软件 yum -y install gcc make python-devel libxml2-devel ...