【BZOJ】2434: [Noi2011]阿狸的打字机
题意
给你一些字符串。\(m\)次询问,每一次询问第\(x\)个字符串在\(y\)字符串中出现了多少次。(输入总长$ \le 10^5$, \(M \le 10^5\))
分析
在ac自动机上,\(x\)字符串出现的所有位置就是其它节点的fail树上有这个节点的节点。即fail树中,\(x\)字符串终止节点的子树。
题解
根据分析,我们只要构造ac自动机和fail树,按dfs序依次便历ac自动机,将节点到根的路径打上标记。每到一个终止节点,就更新以这个节点为模板的匹配串的答案,即这些节点子树的和。
#include <bits/stdc++.h>
using namespace std;
const int N=1e5+10;
char buf[20*N], *is=buf;
int ihead[N], cnt, last[N], c[N][26], fail[N], pos[N], FF[N], LL[N], tot, su[N], ans[N];
inline bool isdig(const char &c) {
return c>='0'&&c<='9';
}
inline bool isok(const char &c) {
return (c>='a'&&c<='z')||c=='P'||c=='B';
}
inline int getint() {
register int x=0;
for(; !isdig(*is); ++is);
for(; isdig(*is); ++is) {
x=x*10+*is-48;
}
return x;
}
inline void putint(int x) {
if(x==0) {
*is++='0';
}
else {
static int s[10], top;
for(top=0; x; x/=10) s[++top]=x%10;
while(top) {
*is++=s[top--]+'0';
}
}
*is++='\n';
}
struct E {
int next, to, id;
}e[N];
inline void add(int x, int y, int id=-1) {
e[++cnt]=(E){ihead[x], y, id}; ihead[x]=cnt;
}
inline void adds(int x, int s) {
for(; x<=tot; x+=x&-x) {
su[x]+=s;
}
}
inline int sum(int x) {
int y=0;
for(; x; x-=x&-x) {
y+=su[x];
}
return y;
}
inline void bfs() {
static int q[N], fr, ta;
fr=ta=0;
q[ta++]=0;
while(fr!=ta) {
int x=q[fr++];
for(int ch=0; ch<26; ++ch) {
if(c[x][ch]) {
int y=c[x][ch];
q[ta++]=y;
if(x==0) {
add(0, y);
continue;
}
fail[y]=c[fail[x]][ch];
add(fail[y], y);
}
else {
c[x][ch]=c[fail[x]][ch];
}
}
}
}
inline void dfs(int x) {
static int fid=0;
FF[x]=++fid;
for(int i=ihead[x]; i; i=e[i].next) {
dfs(e[i].to);
}
LL[x]=fid;
}
inline void getans(int x) {
adds(FF[x], 1);
for(int i=ihead[x]; i; i=e[i].next) {
int y=e[i].to;
ans[e[i].id]=sum(LL[y])-sum(FF[y]-1);
}
for(int ch=0; ch<26; ++ch) {
if(c[x][ch] && (x==0 || c[x][ch]!=c[fail[x]][ch])) {
getans(c[x][ch]);
}
}
adds(FF[x], -1);
}
void init() {
fread(buf, 1, sizeof buf, stdin);
for(; !isok(*is); ++is);
int now=0, n=0, len=0;
for(; isok(*is); ++is) {
if(*is=='P') {
pos[++n]=now;
}
else if(*is=='B') {
now=last[len--];
}
else {
int ch=*is-'a';
if(!c[now][ch]) {
c[now][ch]=++tot;
}
last[++len]=now;
now=c[now][ch];
}
}
bfs();
dfs(0);
}
int main() {
init();
memset(ihead, 0, sizeof(int)*(tot+1));
cnt=0;
int m=getint();
for(int i=0; i<m; ++i) {
int x, y;
x=getint();
y=getint();
add(pos[y], pos[x], i);
}
++tot;
getans(0);
is=buf;
for(int i=0; i<m; ++i) {
putint(ans[i]);
}
fwrite(buf, 1, sizeof(char)*(is-buf), stdout);
return 0;
}
【BZOJ】2434: [Noi2011]阿狸的打字机的更多相关文章
- BZOJ 2434: [Noi2011]阿狸的打字机 [AC自动机 Fail树 树状数组 DFS序]
2434: [Noi2011]阿狸的打字机 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 2545 Solved: 1419[Submit][Sta ...
- BZOJ 2434: [Noi2011]阿狸的打字机( AC自动机 + DFS序 + 树状数组 )
一个串a在b中出现, 那么a是b的某些前缀的后缀, 所以搞出AC自动机, 按fail反向建树, 然后查询(x, y)就是y的子树中有多少是x的前缀. 离线, 对AC自动机DFS一遍, 用dfs序+树状 ...
- bzoj 2434 [Noi2011]阿狸的打字机 AC自动机
[Noi2011]阿狸的打字机 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 4001 Solved: 2198[Submit][Status][D ...
- 【刷题】BZOJ 2434 [Noi2011]阿狸的打字机
Description 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机.打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P'两个字母. 经阿狸研究发现,这个打字机是这样工作的 ...
- BZOJ 2434: [Noi2011]阿狸的打字机 AC自动机+fail树+线段树
Description 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机.打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P'两个字母. 经阿狸研究发现,这个打字机是这样工作的 ...
- bzoj 2434 [Noi2011]阿狸的打字机(fail树+离线处理+BIT)
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2434 [题意] 按照一定规则生成n个字符串,回答若干个询问:(x,y),问第x个字符串 ...
- BZOJ 2434 [Noi2011]阿狸的打字机(AC自动机)
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2434 [题目大意] 给出一个打印的过程,'a'-'z'表示输入字母,P表示打印该字符串 ...
- bzoj 2434 [Noi2011]阿狸的打字机——AC自动机
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2434 dfs AC自动机,走过的点权值+1,回溯的时候权值-1:走到询问的 y 串的节点,看 ...
- bzoj 2434: [Noi2011]阿狸的打字机
#include<cstdio> #include<iostream> #include<cstring> #define M 100008 using names ...
- ●BZOJ 2434: [Noi2011]阿狸的打字机
●赘述题目 (题意就不赘述了) ●解法: ●我先想的一个比较暴力的方法(要TLE): (ac自动机)先求出last数组(参见刘汝佳的解释:last[j]:表示j节点沿着失配指针往回走时,遇到的下一个单 ...
随机推荐
- 菜鸟学Linux命令:nohup命令启动程序
在UNIX/LINUX中,普通进程用&符号放到后台运行,如果启动该程序的控制台logout,则该进程随即终止. 要实现守护进程,一种方法是按守护进程的规则去编程,比较麻烦:另一种方法是仍然用普 ...
- 让那些为Webkit优化的网站也能适配IE10(转载)
转载地址:http://www.w3cplus.com/css3/adapting-your-webkit-optimized-site-for-internet-explorer-10.html 特 ...
- sdut 2610:Boring Counting(第四届山东省省赛原题,划分树 + 二分)
Boring Counting Time Limit: 3000ms Memory limit: 65536K 有疑问?点这里^_^ 题目描述 In this problem you a ...
- [javascript] 使用闭包编写模块
这是一篇[javascript the good parts]的读书笔记. 我们知道可以利用javascript 的prototype 特性为原始类型编写拓展模块.利用如下方法: Object.pro ...
- js特效
1.轮播换图 <!DOCTYPE html> <html lang="en"> <head> <meta charset="UT ...
- ASP.NET 5探险(7):使用混合型控制器方便实现单页应用
(此文章同时发表在本人微信公众号"dotNET每日精华文章",欢迎右边二维码来关注.) 题记:由于在ASP.NET 5中,MVC和WEB API的技术栈合并了,所以开发混合型Con ...
- Linux系统安装及初始化(ubuntu14.04)
Windows 7下硬盘安装Ubuntu 14.04图文教程 Ubuntu 官方已经发布了正式版的 Ubuntu 14.04 LTS,并宣称这是为云计算准备的版本.该版本在云平台和伸缩环境的可靠性.性 ...
- 注解:【无连接表的】Hibernate单向1->N关联
Person与Address关联:单向1->N,[无连接表的] (性能较低,不推荐使用!) Person.java package org.crazyit.app.domain; import ...
- 在Asp.Net MVC中实现RequiredIf标签对Model中的属性进行验证
在Asp.Net MVC中可以用继承ValidationAttribute的方式,自定制实现RequiredIf标签对Model中的属性进行验证 具体场景为:某一属性是否允许为null的验证,要根据另 ...
- 批量删除SharePoint 2010的List中的item
第一种方式:循环遍历List中的所有item,然后根据条件去判断当前item是否应该被删除[注:要用 i-- 方式去遍历,这也是删除集合里面item的常用做法,如果用 i++ 的方式去遍历删除,会出错 ...