Portal

Description

首先给出一个只包含小写字母和'B'、'P'的操作序列\(s_0(|s_0|\leq10^5)\)。初始时我们有一个空串\(t\),依次按\(s_0\)的每一位进行操作:

  • 如果是小写字母,则在\(t\)后面加入这个小写字母;
  • 如果是'B',则删除\(t\)的最后一位;
  • 如果是'P',则复制t到集合\(S\)中。

操作结束后,集合中有\(n(n\leq10^5)\)个字符串,将它们按加入集合的顺序标号为\(1..n\)。接下来\(m(m\leq10^5)\)次询问,每次询问串\(x\)在串\(y\)中出现了几次。

Solution

首先根据\(s_0\)我们可以方便的建出一棵Trie树并建立fail指针,记录代表串\(x\)的节点为\(end[x]\)。然后我们就得到了一个fail树:



一个节点在fail树上的祖先就是它的一个后缀,子树就是以该节点作为后缀的串。那么询问就相当于“求\(end[x]\)的子树中,有多少个点在\(root\)到\(end[y]\)的路径上”。

我们求出fail树的DFS序,然后将询问按\(y\)排序。用树状数组维护每个点是否被标记,当\(y\)转移到\(y+1\)时,按照建立Trie树的方法转移。求\(end[x]\)的子树中有多少个被标记的点就相当于求DFS序的区间和。

时间复杂度\(O(nlogn+mlogn)\)。

Code

//[NOI2011]阿狸的打字机
#include <algorithm>
#include <cstdio>
using std::sort;
int const N=1e5+10;
int n; char s0[N];
struct query{int id,x,y,ans;} q[N];
bool cmpY(query x,query y) {return x.y<y.y;}
bool cmpID(query x,query y) {return x.id<y.id;}
int rt,ndCnt,fa[N],ch[N][26],fail[N]; int end[N];
int Q[N],op,cl;
int edCnt,h[N];
struct edge{int v,nxt;} ed[N];
void edAdd(int u,int v)
{
fail[v]=u;
edCnt++; ed[edCnt].v=v,ed[edCnt].nxt=h[u],h[u]=edCnt;
}
void bldFail()
{
for(int i=0;i<26;i++) ch[0][i]=rt;
Q[++cl]=rt;
while(op<cl)
{
int p=Q[++op];
for(int i=0;i<26;i++)
{
int q=ch[p][i];
if(!q) ch[p][i]=ch[fail[p]][i];
else edAdd(ch[fail[p]][i],q),Q[++cl]=q;
}
}
}
int dfCnt,fr[N],to[N];
void dfs(int u)
{
dfCnt++; fr[u]=dfCnt;
for(int i=h[u];i;i=ed[i].nxt) dfs(ed[i].v);
to[u]=dfCnt;
}
int tr[N];
void add(int x,int v) {while(x<=ndCnt) tr[x]+=v,x+=x&(-x);}
int sum(int x) {int r=0; while(x) r+=tr[x],x-=x&(-x); return r;}
int main()
{
scanf("%s",s0+1);
rt=++ndCnt;
for(int i=1,p=rt;s0[i];i++)
{
int x=s0[i]-'a';
if(s0[i]=='B') p=fa[p];
else if(s0[i]=='P') end[++n]=p;
else {if(!ch[p][x]) fa[ch[p][x]=++ndCnt]=p; p=ch[p][x];}
}
bldFail(); for(int i=1;i<=ndCnt;i++) if(!fr[i]) dfs(i);
int m; scanf("%d",&m);
for(int i=1;i<=m;i++) scanf("%d%d",&q[i].x,&q[i].y),q[i].id=i;
sort(q+1,q+m+1,cmpY);
int now=0,p=rt;
for(int i=0,owo=1,no;owo<=m;owo++)
{
int x=q[owo].x,y=q[owo].y;
while(now<y)
{
i++;
if(s0[i]=='B') add(fr[p],-1),p=fa[p];
else if(s0[i]=='P') now++;
else p=ch[p][s0[i]-'a'],add(fr[p],1);
}
q[owo].ans=sum(to[end[x]])-sum(fr[end[x]]-1);
}
sort(q+1,q+m+1,cmpID);
for(int i=1;i<=m;i++) printf("%d\n",q[i].ans);
return 0;
}

P.S.

双倍经验BZOJ2434

洛谷P2414 - [NOI2011]阿狸的打字机的更多相关文章

  1. 洛谷 P2414 [NOI2011]阿狸的打字机 解题报告

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

  2. 【AC自动机】【树状数组】【dfs序】洛谷 P2414 [NOI2011]阿狸的打字机 题解

        这一题是对AC自动机的充分理解和树dfs序的巧妙运用. 题目背景 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机. 题目描述 打字机上只有28个按键,分别印有26个小写英文字母和' ...

  3. 洛谷P2414 [NOI2011]阿狸的打字机(AC自动机)

    传送门 考虑一下,如果串B在串A中出现过,那么A的fail指针必定直接或间接指向B 那么我们可以把fail树建起来,那么就变成B代表的节点的子树里有多少节点属于A 然后这就是一个序列统计问题,直接用d ...

  4. P2414 [NOI2011]阿狸的打字机

    P2414 [NOI2011]阿狸的打字机 AC自动机+树状数组 优质题解 <------题目分析 先AC自动机搞出Trie图 然后根据fail指针建一只新树 把树映射(拍扁)到一个序列上,用树 ...

  5. BZOJ 2434 Luogu P2414 [NOI2011]阿狸的打字机 (AC自动机、树状数组)

    题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=2434 题解: 我写的是离线做法,不知道有没有在线做法. 转化一波题意,\(x\)在AC ...

  6. P2414 [NOI2011]阿狸的打字机 AC自动机

    题意 给定n个模式串,有m个询问,每次询问第X个模式串在第Y个模中出现了多少次 解题思路 以fail树相反的方向建一棵树T,问题转化为X的子树中有多少个y的终止节点.跑出T的dfs序,X的子树就可以表 ...

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

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

  8. BZOJ 2434: [Noi2011]阿狸的打字机 [AC自动机 Fail树 树状数组 DFS序]

    2434: [Noi2011]阿狸的打字机 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 2545  Solved: 1419[Submit][Sta ...

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

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

随机推荐

  1. PowerShell和Bash的介绍

    PowerShell是运行在windows平台的脚本,而Bash是运行在linux平台的脚本 现在bash能做的事情,PowerShell也能做,PowerShell的强大之处是它可以管理window ...

  2. ping localhost 返回 ::1的导致不能打开http://localhost的原因及解决

    虽然可以在浏览器中正常访问http://localhost但用file,file_get_contents等函数打开http://localhost异常.用127.0.0.1也可以打开,本地hosts ...

  3. P1179 数字统计

    题目描述 请统计某个给定范围[L, R]的所有整数中,数字 2 出现的次数. 比如给定范围[2, 22],数字 2 在数 2 中出现了 1 次,在数 12 中出现 1 次,在数 20 中出 现 1 次 ...

  4. 《Python基础教程》 读书笔记 第六章 抽象 函数 参数

    6.1创建函数 函数是可以调用(可能包含参数,也就是放在圆括号中的值),它执行某种行为并且返回一个值.一般来说,内建的callable函数可以用来判断函数是否可调用: >>> x=1 ...

  5. Failure to transfer org.apache.maven.plugins:maven-compiler-plugin:jar:2.5.1

    Mac上写了一段基于Maven的java代码. 上传Git后,在windows上pull下来,eclipse里面各种错误. ArtifactTransferException:Failure to t ...

  6. 使用laravel的Command实现搜索引擎索引和模板的建立

    创建command,初始化es 创建成功后,可通过php artisan 查看到 php artisan make:command ESInit 安装guzzle composer require g ...

  7. 关于bin和obj文件夹。debug 和release的区别(转)

    关于bin和obj文件夹. 楼主hcaihao(影子男孩)2002-05-29 20:04:24 在 .NET技术 / C# 提问 VS.Net会生成bin和obj文件夹以及它们下面的Debug和Re ...

  8. mac homebrew安装

    http://book.51cto.com/art/201107/278761.htm 3.2.3 使用 Homebrew 安装 Git Mac OS X 有好几个包管理器,用于管理一些开源软件在 M ...

  9. iview upload 上传图片 不传服务器 转 base64

    开始的时候 找不到this了,后来想起来要用 ES6的箭头函数 就有this了 reader.onload = e => { // 读取到的图片base64 数据编码 将此编码字符串传给后台即可 ...

  10. 【软件构造】第三章第四节 面向对象编程OOP

    第三章第四节 面向对象编程OOP 本节讲学习ADT的具体实现技术:OOP Outline OOP的基本概念 对象 类 接口 抽象类 OOP的不同特征 封装 继承与重写(override) 多态与重载( ...