难不成是我后缀自动机学魔怔了,AC 自动机都能套上线段树

题意:给你一颗 Trie,每次询问两个节点 \(u,v\),\(u\) 代表的字符串在 \(v\) 代表的字符串中出现了多少次。

让我们思考一下字符串在 AC 自动机 上是如何匹配的:

  1. 跳儿子节点
  2. 对于每个儿子节点跳 fail,若跳到匹配串就令 ans++

差不多就是标记一个节点 \(u\),然后询问是否有若干个节点在该节点的子树内。

于是,当 \(x\) 相同时,就可以标记 \(x\) 的每一个前缀后,在 fail 树上做一个子树和。查询的时候直接查询子树和就好了。

\(x\) 不定的时候用线段树来维护就好啦。

不过重要的一点是:如何维护一个节点是哪些字符串的前缀。

我们发现,每次 \(B\) 跳 father 的时候,经过这个节点的字符串是连续的一段。相当于使用线段树区间修改。所以标记一下就好啦。

需要注意一点是线段树要标记永久化。

code:

#include<cstring>
#include<cstdio>
#include<queue>
typedef unsigned uint;
const uint M=1e5+5;
uint n,m,cnt,tot,len,fa[M],pos[M],lst[M],fail[M],root[M],chi[M][26];char s[M];
uint L,R,q[M];
struct Node{
uint L,R,tag;
}t[M*50];
inline uint min(const uint&a,const uint&b){
return a-b>>31?a:b;
}
inline uint max(const uint&a,const uint&b){
return a-b>>31?b:a;
}
void Modify(uint&u,const uint&l,const uint&r,const uint&L=1,const uint&R=len){
if(l>R||L>r)return;if(!u)u=++cnt;
if(l<=L&&R<=r)return void(t[u].tag=1);
uint mid=L+R>>1;
Modify(t[u].L,l,r,L,mid);Modify(t[u].R,l,r,mid+1,R);
}
uint Query(const uint&u,const uint&x,const uint&L=1,const uint&R=len){
if(!u)return 0;if(L==R)return t[u].tag;
uint mid=L+R>>1;
if(x<=mid)return Query(t[u].L,x,L,mid)+t[u].tag;
else return Query(t[u].R,x,mid+1,R)+t[u].tag;
}
uint Merge(const uint&q,const uint&p){
if(!q||!p)return q|p;
uint u=++cnt;
t[u].L=Merge(t[q].L,t[p].L);
t[u].R=Merge(t[q].R,t[p].R);
t[u].tag=t[q].tag+t[p].tag;
return u;
}
inline void Build(){
uint i,u,c;L=1;
for(c=0;c^26;++c)if(chi[1][c])fail[q[++R]=chi[1][c]]=1;
while(L<=R){
if(!fail[u=q[L++]])fail[u]=1;
for(c=0;c^26;++c){
if(chi[u][c])fail[q[++R]=chi[u][c]]=chi[fail[u]][c];
else chi[u][c]=chi[fail[u]][c];
}
}
}
inline void init(){
uint i,u=++tot,now=1;n=strlen(s);
for(i=0;i<n;++i){
if(s[i]=='B')u=fa[u];
else if(s[i]=='P')++len;
else{
if(!chi[u][s[i]-97])fa[chi[u][s[i]-97]=++tot]=u;
u=chi[u][s[i]-97];
}
}
u=1;Build();
for(i=0;i<n;++i){
if(s[i]=='B')lst[u]^now?Modify(root[u],lst[u],now-1):void(),u=fa[u];
else if(s[i]=='P')pos[now++]=u;
else lst[u=chi[u][s[i]-97]]=now;
}
while(u^1)lst[u]^now?Modify(root[u],lst[u],now-1):void(),u=fa[u];
do root[fail[q[R]]]=Merge(root[fail[q[R]]],root[q[R]]);while(--R);
}
signed main(){
uint x,y;scanf("%s%u",s,&m);init();
while(m--)scanf("%u%u",&x,&y),printf("%u\n",Query(root[pos[x]],y));
}

LGP2414题解的更多相关文章

  1. 2016 华南师大ACM校赛 SCNUCPC 非官方题解

    我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...

  2. noip2016十连测题解

    以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...

  3. BZOJ-2561-最小生成树 题解(最小割)

    2561: 最小生成树(题解) Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1628  Solved: 786 传送门:http://www.lyd ...

  4. Codeforces Round #353 (Div. 2) ABCDE 题解 python

    Problems     # Name     A Infinite Sequence standard input/output 1 s, 256 MB    x3509 B Restoring P ...

  5. 哈尔滨理工大学ACM全国邀请赛(网络同步赛)题解

    题目链接 提交连接:http://acm-software.hrbust.edu.cn/problemset.php?page=5 1470-1482 只做出来四道比较水的题目,还需要加强中等题的训练 ...

  6. 2016ACM青岛区域赛题解

    A.Relic Discovery_hdu5982 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Jav ...

  7. poj1399 hoj1037 Direct Visibility 题解 (宽搜)

    http://poj.org/problem?id=1399 http://acm.hit.edu.cn/hoj/problem/view?id=1037 题意: 在一个最多200*200的minec ...

  8. 网络流n题 题解

    学会了网络流,就经常闲的没事儿刷网络流--于是乎来一发题解. 1. COGS2093 花园的守护之神 题意:给定一个带权无向图,问至少删除多少条边才能使得s-t最短路的长度变长. 用Dijkstra或 ...

  9. CF100965C题解..

    求方程 \[ \begin{array}\\ \sum_{i=1}^n x_i & \equiv & a_1 \pmod{p} \\ \sum_{i=1}^n x_i^2 & ...

随机推荐

  1. 学习jsp篇:jsp简单实例之二登录

    编程环境:IDEA,Tomcat,JavaEE 一.实例二登录 1.在自己建的工程下的web目录下建一个文件夹为login,在login中编写登录代码(其实就是和实例一同一个项目) 2.先建登录页面j ...

  2. CocoaPods使用专题 by h.l

    cocoaPods安装 CocoaPods安装和使用教程(code4app) cocoapods使用问题解决 cocoapods慢如何解决? CocoaPods停在Analyzing dependen ...

  3. OSPF路由协议基础知识

    OSPF路由协议 1.OSPF的基本概念 2.OSPF邻接关系的建立 3.OSPF的应用环境 4.OSPF的基本配置命令 1.OSPF区域为了适应大型的网络,OSPF在AS(自治系统)内划分多个区域. ...

  4. kubectl详解

    kubectl详解 目录 kubectl详解 一.陈述式管理 1. 陈述式资源管理方法 2. k8s相关信息查看 2.1 查看版本信息 2.2 查看资源对象简写 2.3 查看集群信息 2.4 配置ku ...

  5. go基础——变量与常量

    变量 package main import "fmt" /* 变量:variable 概念:一小块内存,用于存储数据,在程序运行过程中数值可以改变 特性:静态语言,强类型语言 * ...

  6. 系统操作命令实践 下(系统指令+增删改查+vim编辑器)

    目录 1.考试 2.今日问题 3.今日内容 4.复制文件 4.移动文件 Linux文件查看补充 cat , nl 5.删除文件 6.系统别名 7.vi/vim编辑器 系统操作命令实践 下(系统指令+增 ...

  7. Solution -「HDU #6566」The Hanged Man

    \(\mathcal{Description}\)   Link.   给定一棵含 \(n\) 个点的树,每个结点有两个权值 \(a\) 和 \(b\).对于 \(k\in[1,m]\),分别求 \[ ...

  8. Spring中publish如何将多个Event和多个Listener进行无误差匹配

    从命令模式的维度理解Spring 之Application Event - 池塘里洗澡的鸭子 - 博客园 (cnblogs.com)中通自定义简单的Event和Listener阐述了Spring中pu ...

  9. 『无为则无心』Python面向对象 — 53、对Python中封装的介绍

    目录 1.继承的概念 2.继承的好处 3.继承体验 4.单继承 5.多继承 1.继承的概念 在Python中,如果两个类存在父子级别的继承关系,子类中即便没有任何属性和方法,此时创建一个子类对象,那么 ...

  10. RENIX流量发送模式——网络测试仪实操

    信而泰的RENIX平台支持5种传输模式, 如下图所示. 模式一: continuous Continuous:连续发送流 持续发送方式是默认的发送模式,  它不会自动的停止, 会一直发送, 直到用户手 ...