题意

有一个打字机,支持三种操作:

  • 字符串末尾加一个小写字母
  • 字符串末尾减一个字符
  • 输出这个字符串

经过不超过\(n\)次操作后有\(m\)组询问:\((x,y)\),表示第\(x\)次输出第字符串在第\(y\)次输出第字符串里出现几次

\(n,m \leq 10^5\)

题解

每次加减字符就在trie树上走,输出的话记录一下在哪个结点

然后考虑询问\((x,y)\)暴力怎么做:\(x\)应该是\(y\)一个前缀的后缀,于是我们对于从根到\(y\)路径上每个结点(这相当于枚举\(y\)的后缀),从这个结点跳\(fail\),如果跳到\(x\)就\(ans++\),然后考虑下一个结点

实际上我们要求的就是根到\(y\)这条链上的结点中,在\(fail\)树中是\(x\)儿子的个数

我们可以按\(\text{trie}\)树的\(\text{dfs}\)序枚举\(y\),这样枚举所有的链和信息是\(O(n)\)的,每个点只会被加入一次和删除一次。然后考虑回答所有\((i,y)\)的询问,直接询问当前在\(x\)的\(fail\)树子树的结点个数。可以使用树状数组维护。具体说就是把询问按\(y\)在\(trie\)上的\(\text{dfs}\)序排序,然后每个点必须插入到它\(fail\)树\(\text{dfs}\)序的位置,查询就找到\(x\)的\(fail\)子树的\(\text{dfs}\)区间进行查询。

实现的话注意\(trie\)和\(fail\)不要搞混了,另外这题可以用主席树在线做

#include <algorithm>
#include <cstdio>
#include <vector>
using namespace std; const int N = 2e5 + 10; int ch[N][26], fa[N], fail[N];
int dfn[N], dl[N], dr[N], dn[N];
int pos = 1, id = 1, n, pt[N], ans[N];
vector<int> fs[N]; struct qs {
int x, y, id;
bool operator < (const qs &b) const {
return dfn[y] < dfn[b.y];
}
} q[N]; void work(char c) {
if(c == 'B') pos = fa[pos];
else if(c == 'P') pt[++ pt[0]] = pos;
else {
int &v = ch[pos][c - 'a'];
if(!v) {
v = ++ id;
fa[v] = pos;
}
pos = v;
}
} void dfs(int u) { //on trie
dfn[u] = ++ dfn[0]; dn[dfn[0]] = u;
for(int i = 0; i < 26; i ++)
if(ch[u][i]) dfs(ch[u][i]);
} void buildac() {
static int q[N], l, r, v;
for(int i = 0; i < 26; i ++) if(v = ch[1][i]) {
q[r ++] = v; fail[v] = 1;
} else ch[1][i] = 1;
while(l < r) {
int u = q[l ++];
for(int i = 0; i < 26; i ++) if(v = ch[u][i]) {
q[r ++] = v; fail[v] = ch[fail[u]][i];
} else ch[u][i] = ch[fail[u]][i];
}
for(int i = 2; i <= id; i ++)
fs[fail[i]].push_back(i);
} void dfs2(int u) { //on fail tree
dl[u] = ++ dl[0];
for(int i = 0; i < fs[u].size(); i ++) dfs2(fs[u][i]);
dr[u] = dl[0];
} int bit[N]; void add(int x, int y) {
for(; x <= id; x += x & (-x)) bit[x] += y;
} int qry(int x) {
int ans = 0;
for(; x >= 1; x &= x - 1) ans += bit[x];
return ans;
} int main() {
static char s[N]; scanf("%s", s);
for(char *c = s; *c; c ++) work(*c);
dfs(1); buildac(); dfs2(1);
scanf("%d", &n);
for(int i = 1; i <= n; i ++) {
scanf("%d%d", &q[i].x, &q[i].y);
q[i].x = pt[q[i].x];
q[i].y = pt[q[i].y]; //id -> node
q[i].id = i;
}
sort(q + 1, q + n + 1);
for(int i = 1, j = 1; i <= id; i ++) {
int u = dn[i];
if(i > 1) {
int la = dn[i - 1];
while(la != fa[u]) {
add(dl[la], -1);
la = fa[la];
}
}
add(dl[u], 1);
for(; j <= n && dfn[q[j].y] == i; j ++) {
ans[q[j].id] = qry(dr[q[j].x]) - qry(dl[q[j].x] - 1);
}
}
for(int i = 1; i <= n; i ++)
printf("%d\n", ans[i]);
return 0;
}

「BZOJ 2434」「NOI 2011」阿狸的打字机「AC自动机」的更多相关文章

  1. 【BZOJ 2434】【NOI 2011】阿狸的打字机 fail树

    完全不会啊,看题解还看了好久,我是蒟蒻$QAQ$ $zyf$的题解挺好的:http://blog.csdn.net/clove_unique/article/details/51059425 $fai ...

  2. BZOJ 2434 [Noi2011]阿狸的打字机(AC自动机)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2434 [题目大意] 给出一个打印的过程,'a'-'z'表示输入字母,P表示打印该字符串 ...

  3. NOI 2011 阿狸的打字机(AC自动机+主席树)

    题意 https://loj.ac/problem/2444 思路 ​多串匹配,考虑 \(\text{AC}\) 自动机.模拟打字的过程,先建出一棵 \(\text{Trie}\) 树,把它变成自动机 ...

  4. 【NOI 2011】阿狸的打字机

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

  5. NOI 2011 【阿狸的打字机】

    之前讲了[AC自动姬],今天我终于把这题给刚下来了...嗯,来给大家讲一讲. 题目描述: 打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P'两个字母.经阿狸研究发现,这个打字机是这样工 ...

  6. 「AC自动机」学习笔记

    AC自动机(Aho-Corasick Automaton),虽然不能够帮你自动AC,但是真的还是非常神奇的一个数据结构.AC自动机用来处理多模式串匹配问题,可以看做是KMP(单模式串匹配问题)的升级版 ...

  7. 「BZOJ 4228」Tibbar的后花园

    「BZOJ 4228」Tibbar的后花园 Please contact lydsy2012@163.com! 警告 解题思路 可以证明最终的图中所有点的度数都 \(< 3\) ,且不存在环长是 ...

  8. 「BZOJ 3645」小朋友与二叉树

    「BZOJ 3645」小朋友与二叉树 解题思路 令 \(G(x)\) 为关于可选大小集合的生成函数,即 \[ G(x)=\sum[i\in c ] x^i \] 令 \(F(x)\) 第 \(n\) ...

  9. 「BZOJ 4502」串

    「BZOJ 4502」串 题目描述 兔子们在玩字符串的游戏.首先,它们拿出了一个字符串集合 \(S\),然后它们定义一个字符串为"好"的,当且仅当它可以被分成非空的两段,其中每一段 ...

随机推荐

  1. js获取用户实时地理位置

    js获取用户实时地理位置 if(navigator.geolocation) { var id = navigator.geolocation.watchPosition(function(posit ...

  2. 1.2celery的安装

    我们可以使用python的包管理器pip来安装: pip install -U Celery 也可从官方直接下载安装包:https://pypi.python.org/pypi/celery/ tar ...

  3. MySQL: [Err] 1093 - You can't specify target table 'bk' for update in FROM clause

    错误的意思说,不能先select出同一表中的某些值,再update这个表(在同一语句中). 例如下面这个sql: delete from tbl where id in (        select ...

  4. 微信公众号php从0开发,包括功能(自定义菜单,分享)

    之前写的一篇微信公众号文章. 工作需要,进行此次调研,并记录开发过程. 开发目的,页面授权,页面获取用户头像,用户昵称 微信id, 分享页面. 微信订阅号 无法获取用户个人信息 写在记录前,公众号也是 ...

  5. Python模块及其导入

    一.模块 1.模块的定义: 为了编写可维护的代码,我们把很多函数分组,分别放到不同的文件里,这样,每个文件包含的代码就相对较少, 很多编程语言都采用这种组织代码的方式.在Python中,一个.py文件 ...

  6. eclipse egit(分支管理 上)

    这一章比较重要,讲述了Git比svn强大的地方,直接转载廖雪峰老师的文字,更好的理解 什么是分支 和 为什么分支git比svn做的更好 分支在实际中有什么用呢?假设你准备开发一个新功能,但是需要两周才 ...

  7. tomcat报503 或者无法启动应用

    一般都是配置文件有问题,或者路径问题,或者jvm的参数路径问题.... 总之,报错实在是不清楚!这点比resin差远了!!

  8. FZU2282 Wand

    题意 n个数字,要求至少k个数字位置不变,其余进行错排的方案数 分析 错排公式: D(n)=(n-1)[D(n-2)+D(n-1)]  如果n个数字,i个数字位置不变,其余进行错排的的方案数是C(n, ...

  9. Gym - 101128C:Canvas Painting

    这个就是哈夫曼树哇~ 我们仨英语太差了,跟榜时候才看出来是哈夫曼树雾 一个优先队列就可以搞定 #include <cstdio> #include <algorithm> #i ...

  10. jquery动画效果总结

    一.jquery的动画速度fast-slow对应着600ms,200ms,不写是400ms 二.禁用jquery的动画 将jQuery.fx.off设置为true即可 $(".stoppin ...