题意

给你一些串,还有一些询问

问你第x个串在第y个串中出现了多少次

思路

对这些串建ac自动机

根据fail树的性质:若x节点是trie中root到t任意一个节点的fail树的祖先,那么x一定是y的子串

而x在y中出现的次数为以x为fail树中的根节点的子树中,有多少个节点是trie树中根节点到y的

首先对询问离线

由于这题是一个节点一个节点建的ac自动机,所以我们可以根据这个建立的路径来维护一个当前节点到根的路径

路径上的点权为1,其余为0

每次到达一个询问到的y,我们就查询此时x的fail树中的子树和,这个可以预处理出fail树的dfs序用树状数组维护

但是如果排除这题的特殊性,要通过遍历trie的方式来更新答案

因为在操作fail的过程中,trie改变了

else trie[now][i]=trie[fail[now]][i];

也就是说bfs构造fail指针的时候,有时候会让trie指向之前的节点,也就是(trie中)高度比较小的节点

所以dfs只需要这样写

void dfs(int x){
//printf("==%d\n",x);
add(bg[x],1);
for(int i = 0; i < (int)Q[x].size(); i++){
int y = Q[x][i].fst;
ans[Q[x][i].sc]=sum(ed[y])-sum(bg[y]-1);
}
for(int i = 0; i < 26; i++){
int y = ac.trie[x][i];
if(!y||h[y]<=h[x])continue;
dfs(y);
}
add(bg[x],-1);
}

代码

版本一

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cmath>
#include<cstring>
#include<string>
#include<stack>
#include<queue>
#include<deque>
#include<set>
#include<vector>
#include<map>
#include<functional> #define fst first
#define sc second
#define pb push_back
#define mem(a,b) memset(a,b,sizeof(a))
#define lson l,mid,root<<1
#define rson mid+1,r,root<<1|1
#define lc root<<1
#define rc root<<1|1 using namespace std; typedef double db;
typedef long double ldb;
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> PI;
typedef pair<ll,ll> PLL; const db eps = 1e-6;
const int mod = 1e9+7;
const int maxn = 3e5+100;
const int maxm = 2e6+100;
const int inf = 0x3f3f3f3f;
const db pi = acos(-1.0); vector<int>v[maxn];
int id[maxn];
struct AC{
//局部变量没有默认0!
int trie[maxn][26];
int num[maxn];//单词出现次数
int fail[maxn],fa[maxn];
int vis[maxn];//ask函数用到
int tot,rt;
int ntot;
//多测可写个init
void init(){tot=0;mem(vis,0);mem(trie,0);rt=0;ntot=0;}
void add(char c){
if(c=='P'){
num[rt]++;
id[++ntot]=rt;
return;
}
if(c=='B'){
rt=fa[rt];
return;
}
int x = c-'a';
if(!trie[rt][x]){
trie[rt][x]=++tot;
fa[tot]=rt;
}
//fa[trie[rt][x]]=rt;
rt=trie[rt][x];
}
void build(){
queue<int>q;
for(int i = 0; i < 26; i++){
if(trie[0][i]){
//fail[trie[0][i]]=0;
//v[0].pb(trie[0][i]);
q.push(trie[0][i]);
}
}
while(!q.empty()){
int now = q.front();
q.pop();
for(int i = 0; i < 26; i++){
////让这个节点的失败指针指向(((他父亲节点)的失败指针所指向的那个节点)的下一个相同节点)
if(trie[now][i]){
fail[trie[now][i]]=trie[fail[now]][i];
//v[trie[fail[now]][i]].pb(trie[now][i]);
q.push(trie[now][i]);
}
else trie[now][i]=trie[fail[now]][i];
//否则就让当前节点的这个子节点指向当前节点fail指针的这个子节点
}
v[fail[now]].pb(now);
}
} }ac;
char a[maxn];
int bg[maxn],ed[maxn];
int S[maxn],tot;
void gao(int x){
S[++tot]=x;
bg[x]=tot;
for(int i = 0; i < (int)v[x].size(); i++){
int y =v[x][i];
gao(y);
}
//S[++tot]=x;
ed[x]=tot;
}
int tree[maxn];
int lowbit(int x){return x&-x;}
void add(int x, int c){
for(int i = x; i <= tot; i+=lowbit(i)){tree[i]+=c;}
}
int sum(int x){
int ans =0;
for(int i = x; i; i-=lowbit(i))ans+=tree[i];
return ans;
}
vector<PI>Q[maxn];
int ans[maxn];
/*
void dfs(int x){
//printf("%d\n",x);
add(bg[x],1);
for(int i = 0; i < (int)Q[x].size(); i++){
int y = Q[x][i].fst;
ans[Q[x][i].sc]=sum(ed[y])-sum(bg[y]-1);
}
for(int i = 0; i < 26; i++){
int y = ac.trie[x][i];
if(!y||(y==ac.trie[ac.fail[x]][i]))continue;
dfs(y);
}
add(bg[x],-1);
}*/
void sv(){
int now = 0;
int n = strlen(a+1);
int world=0;
for(int i = 1; i <= n; i++){
if(a[i]=='B'){
add(bg[now],-1);
now=ac.fa[now];
continue;
}
else if(a[i]=='P'){
world++;
for(int j = 0; j < (int)Q[now].size(); j++){
int x = Q[now][j].fst;
int y = Q[now][j]. sc;
ans[y]=sum(ed[x])-sum(bg[x]-1);
}
continue;
}
else{
int x=a[i]-'a';
now=ac.trie[now][x];
add(bg[now],1);
}
}
}
int main(){
ac.init();
tot=0;
scanf("%s", a+1);
int n = strlen(a+1);
for(int i = 1; i <= n; i++){
ac.add(a[i]);
}
ac.build();
int m;
scanf("%d", &m);
for(int i = 1; i <= m; i++){
int x,y;
scanf("%d %d", &x, &y);
x=id[x];y=id[y];
Q[y].pb(make_pair(x,i));
}
gao(0);
//dfs(0);
sv();
for(int i = 1; i <= m; i++){
printf("%d\n",ans[i]);
}
return 0;
}
/*
aPaPBbP
3
1 2
1 3
2 3 a
aa
aa
*/

版本二

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cmath>
#include<cstring>
#include<string>
#include<stack>
#include<queue>
#include<deque>
#include<set>
#include<vector>
#include<map>
#include<functional> #define fst first
#define sc second
#define pb push_back
#define mem(a,b) memset(a,b,sizeof(a))
#define lson l,mid,root<<1
#define rson mid+1,r,root<<1|1
#define lc root<<1
#define rc root<<1|1 using namespace std; typedef double db;
typedef long double ldb;
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> PI;
typedef pair<ll,ll> PLL; const db eps = 1e-6;
const int mod = 1e9+7;
const int maxn = 3e5+100;
const int maxm = 2e6+100;
const int inf = 0x3f3f3f3f;
const db pi = acos(-1.0); vector<int>v[maxn];
int id[maxn];
int h[maxn];
struct AC{
//局部变量没有默认0!
int trie[maxn][26];
int num[maxn];//单词出现次数
int fail[maxn],fa[maxn];
int vis[maxn];//ask函数用到
int tot,rt;
int ntot;
int H;
//多测可写个init
void init(){tot=0;mem(vis,0);mem(trie,0);rt=0;ntot=0;H=0;}
void add(char c){
if(c=='P'){
num[rt]++;
id[++ntot]=rt;
return;
}
if(c=='B'){
rt=fa[rt];H--;
return;
}
int x = c-'a';
if(!trie[rt][x]){
trie[rt][x]=++tot;
fa[tot]=rt;
}
//fa[trie[rt][x]]=rt;
rt=trie[rt][x];
h[rt]=++H;
}
void build(){
queue<int>q;
for(int i = 0; i < 26; i++){
if(trie[0][i]){
fail[trie[0][i]]=0;
//v[0].pb(trie[0][i]);
q.push(trie[0][i]);
}
}
while(!q.empty()){
int now = q.front();
q.pop();
for(int i = 0; i < 26; i++){
////让这个节点的失败指针指向(((他父亲节点)的失败指针所指向的那个节点)的下一个相同节点)
if(trie[now][i]){
fail[trie[now][i]]=trie[fail[now]][i];
//v[trie[fail[now]][i]].pb(trie[now][i]);
q.push(trie[now][i]);
}
else trie[now][i]=trie[fail[now]][i];
//否则就让当前节点的这个子节点指向当前节点fail指针的这个子节点
}
v[fail[now]].pb(now);
}
} }ac;
char a[maxn];
int bg[maxn],ed[maxn];
int S[maxn],tot;
void gao(int x){
S[++tot]=x;
bg[x]=tot;
for(int i = 0; i < (int)v[x].size(); i++){
int y =v[x][i];
gao(y);
}
//S[++tot]=x;
ed[x]=tot;
}
int tree[maxn];
int lowbit(int x){return x&-x;}
void add(int x, int c){
for(int i = x; i <= tot; i+=lowbit(i)){tree[i]+=c;}
}
int sum(int x){
int ans =0;
for(int i = x; i; i-=lowbit(i))ans+=tree[i];
return ans;
}
vector<PI>Q[maxn];
int ans[maxn]; void dfs(int x){
//printf("==%d\n",x);
add(bg[x],1);
for(int i = 0; i < (int)Q[x].size(); i++){
int y = Q[x][i].fst;
ans[Q[x][i].sc]=sum(ed[y])-sum(bg[y]-1);
}
for(int i = 0; i < 26; i++){
int y = ac.trie[x][i];
if(!y||h[y]<=h[x])continue;
dfs(y);
}
add(bg[x],-1);
}
void sv(){
int now = 0;
int n = strlen(a+1);
int world=0;
for(int i = 1; i <= n; i++){
if(a[i]=='B'){
add(bg[now],-1);
now=ac.fa[now];
continue;
}
else if(a[i]=='P'){
world++;
for(int j = 0; j < (int)Q[now].size(); j++){
int x = Q[now][j].fst;
int y = Q[now][j]. sc;
ans[y]=sum(ed[x])-sum(bg[x]-1);
}
continue;
}
else{
int x=a[i]-'a';
now=ac.trie[now][x];
add(bg[now],1);
}
}
}
int main(){
ac.init();
tot=0;
scanf("%s", a+1);
int n = strlen(a+1);
for(int i = 1; i <= n; i++){
ac.add(a[i]);
}
ac.build();
int m;
scanf("%d", &m);
for(int i = 1; i <= m; i++){
int x,y;
scanf("%d %d", &x, &y);
x=id[x];y=id[y];
Q[y].pb(make_pair(x,i));
}
gao(0);
dfs(0);
//sv();
for(int i = 1; i <= m; i++){
printf("%d\n",ans[i]);
}
return 0;
}
/*
asPdPasPddPhBdPhPnaPasP
8
1 5
2 6
3 8
4 3
7 7
2 5
6 8
1 8 a
aa
aa
*/

BZOJ 2434 阿狸的打字机(ac自动机+dfs序+树状数组)的更多相关文章

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

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

  2. NOI 2011 阿狸的打字机 (AC自动机+dfs序+树状数组)

    题目大意:略(太长了不好描述) 良心LOJ传送门 先对所有被打印的字符串建一颗Trie树 观察数据范围,并不能每次打印都从头到尾暴力建树,而是每遍历到一个字符就在Trie上插入这个字符,然后记录每次打 ...

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

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

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

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

  5. BZOJ_2434_[NOI2011]_阿狸的打字机_(AC自动机+dfs序+树状数组)

    描述 http://www.lydsy.com/JudgeOnline/problem.php?id=2434 给出\(n\)个字符串,\(m\)个询问,对于第\(i\)个询问,求第\(x_i\)个字 ...

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

    [NOI2011]阿狸的打字机 题目链接:https://www.luogu.org/problemnew/show/P2414 题目背景 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机. ...

  7. CodeForces -163E :e-Government (AC自动机+DFS序+树状数组)

    The best programmers of Embezzland compete to develop a part of the project called "e-Governmen ...

  8. BZOJ_3881_[Coci2015]Divljak_AC自动机+dfs序+树状数组

    BZOJ_3881_[Coci2015]Divljak_AC自动机+dfs序+树状数组 Description Alice有n个字符串S_1,S_2...S_n,Bob有一个字符串集合T,一开始集合是 ...

  9. [NOI2011][bzoj2434] 阿狸的打字机 [AC自动机+dfs序+fail树+树状数组]

    题面 传送门 正文 最暴力的 最暴力的方法:把所有询问代表的字符串跑一遍kmp然后输出 稍微优化一下:把所有询问保存起来,把模板串相同的合并,求出next然后匹配 但是这两种方法本质没有区别,都是暴力 ...

随机推荐

  1. Ceph 文件系统 CephFS 的实战配置,等你来学习 -- <4>

    Ceph 文件系统 CephFS 的介绍与配置 CephFs介绍 Ceph File System (CephFS) 是与 POSIX 标准兼容的文件系统, 能够提供对 Ceph 存储集群上的文件访问 ...

  2. 【Java基础总结】总结

    总想着把学习的过程全都记录下来 以便某一时刻回头的时候,还能看见走过的路 对于基础来说,即使不回头看,也知道这条路是什么样子的 记录不记录,都无所谓 况且我不是专业的记录者,不记录比记录好 实在想不起 ...

  3. navicate远程连接mysql8.0失败

    已经给了远程连接权限(update mysql.user set host = "%" where user = 'root'; flush privileges;) 连接错误提示 ...

  4. Alodi:环境创建从未如此简单

    一个满足你各种想象的快速方便生成临时环境的系统 在『Alodi:为了保密我开发了一个系统』文章中有讲到我们开发了一个系统用来快速生成临时测试环境,短短三个月已有数百个环境被创建,简化了工作,节省了时间 ...

  5. React Hooks 实现和由来以及解决的问题

    与React类组件相比,React函数式组件究竟有何不同? 一般的回答都是: 类组件比函数式组件多了更多的特性,比如 state,那如果有 Hooks 之后呢? 函数组件性能比类组件好,但是在现代浏览 ...

  6. Java 基础(四)| IO 流之使用文件流的正确姿势

    为跳槽面试做准备,今天开始进入 Java 基础的复习.希望基础不好的同学看完这篇文章,能掌握泛型,而基础好的同学权当复习,希望看完这篇文章能够起一点你的青涩记忆. 一.什么是 IO 流? 想象一个场景 ...

  7. Java 基础(三)| IO流之使用 File 类的正确姿势

    为跳槽面试做准备,今天开始进入 Java 基础的复习.希望基础不好的同学看完这篇文章,能掌握泛型,而基础好的同学权当复习,希望看完这篇文章能够起一点你的青涩记忆. 一.什么是 File 类? java ...

  8. django.db.migrations.exceptions.MigrationSchemaMissing和raise ImproperlyConfigured('mysqlclient 1.3.13 or newer is required; you have %s.' % Database.__version__)

    1.使用Python3.7 + Django2.2 + MySQL 5.5 在执行(python manage.py migrate)命令时出现错误django.db.migrations.excep ...

  9. Nmap使用教程(一)

    基本扫描技术 扫描单个网络 nmap 192.168.1.1/www.baidu.com 扫描多个网络/目标 nmap 192.168.1.1 192.168.1.2 #将扫描同个网段内不同的ip地址 ...

  10. CTF实验吧——证明自己吧

    题目地址:http://www.shiyanbar.com/ctf/28 没有壳 ,vc++ 写的 拖进OD观察观察,发现代码很短哟,先来看这俩个call 怀疑他们其中有正确的flag和我们输入的东西 ...