题意

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

问你第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. 傅立叶变换—FFT(cuda实现)

    背景: 无意间看到cuda解决FFT有一个cufft函数库,大体查看了有关cufft有关知识,写了一个解决一维情况的cuda代码,据调查知道cufft在解决1D,2D,3D的情况时间复杂度都为O(nl ...

  2. .sarut后缀病毒,勒索病毒

    前两天朋友的电脑中所有的文件后缀名都被改为.sarut 一看就是中了勒索病毒 每个文件夹下都有一个勒索信 查资料后发现这个病毒是STOP病毒的变种 可能是朋友使用windows激活工具了,然后这个病毒 ...

  3. spring boot集成spring-boot-starter-mail邮件功能

    前情提要 以目前IT系统功能来看,邮件功能是非常重要的一个功能.例如:找回密码.邮箱验证,邮件动态码.忘记密码,邮件营销等,都需要用到邮件功能.结合当下最流行的spring boot微服务,推出了sp ...

  4. 虚拟机下 windows 自动配置 IPv4 地址 169.254.X.X(首选)

    问题: windows server上,自己手动配置的ip不生效,自动获取地址虽然ok,但是服务器必须指定ip. 诊段: ipconfig/all里查看 自动配置 IPv4 地址  169.254.X ...

  5. Java 使用Scanner时的NoSuchElementException异常

    做实验时设计了一个类,在类中的两个不同函数中分别创建了两个Scanner对象,并且在各个函数的结尾使用了close()方法,结果在运行时产生了NoSuchElementException异常. 实验的 ...

  6. AVR单片机教程——小结

    本文隶属于AVR单片机教程系列.   第一期挺让我失望的,是我太菜,没有把想讲的都讲出来.经常写了很多,然后一点一点删掉,最后就没多少了. 而且感觉难度不合适,处于很尴尬的位置.讲得简单,难的丢给库, ...

  7. Excel-条件格式

    今天运用了一下条件格式中的自建规则进行公式筛选, 设置格式那里一定要将$P$8修改为$P8 然后双击修改后的第一项进行单元格的自动填充

  8. CSS盒子模型与双飞翼布局

    盒子模型&双飞翼实现 CSS盒模型本质上是一个盒子,封装周围的HTML元素,它包括:外边距(margin).边框(border).内边距(padding).实际内容(content)四个属性. ...

  9. HTML5浏览器支持及兼容性处理

    1.现代的浏览器都支持HTML5. 2.所有浏览器不管是新的还是旧的对无法识别的元素会作为内联元素自动处理. 3.HTML5定义了8个HTML语义元素,所有这些元素都是块级元素,为了能让旧版本的浏览器 ...

  10. model form

    ModelForm 能允许我们通过一个 Model 直接创建一个和该模型的字段一一对应的表单,大大方便了表单操作. 下面来看一个例子. 首先我们有这样的 model: from django.db i ...