博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
BZOJ2434:[NOI2011]阿狸的打字机——题解
阅读量:6831 次
发布时间:2019-06-26

本文共 1377 字,大约阅读时间需要 4 分钟。

打字机上只有28个按键,分别印有26个小写英文字母和'B'、'P'两个字母。经阿狸研究发现,这个打字机是这样工作的:

·输入小写字母,打字机的一个凹槽中会加入这个字母(这个字母加在凹槽的最后)。

·按一下印有'B'的按键,打字机凹槽中最后一个字母会消失。

·按一下印有'P'的按键,打字机会在纸上打印出凹槽中现有的所有字母并换行,但凹槽中的字母不会消失。

例如,阿狸输入aPaPBbP,纸上被打印的字符如下:

a aa ab 我们把纸上打印出来的字符串从1开始顺序编号,一直到n。打字机有一个非常有趣的功能,在打字机中暗藏一个带数字的小键盘,在小键盘上输入两个数(x,y)(其中1≤x,y≤n),打字机会显示第x个打印的字符串在第y个打印的字符串中出现了多少次。

阿狸发现了这个功能以后很兴奋,他想写个程序完成同样的功能,你能帮助他么?

因为没有前置技能现学了一发fail树,再回头看就是fail树棵题了。

因为询问x串在y串出现几次,可以直接转化为y串在trie的节点在fail树x串末节点的子树中出现了几次(这个比较绕,但是很好想。)

而且因为这题特殊的建trie的方式,我们完全可以离线,对y排序,然后重新跑一遍建trie的过程,这样就保证了y的快(顺)速(序)查找。

那么中间记录y节点的工作就交给树状数组完成即可。

#include
#include
#include
#include
#include
#include
#include
using namespace std;typedef long long ll;const int S=1e5+5;inline int read(){ int X=0,w=0;char ch=0; while(!isdigit(ch)){w|=ch=='-';ch=getchar();} while(isdigit(ch))X=(X<<3)+(X<<1)+(ch^48),ch=getchar(); return w?-X:X;}struct trie{ int a[26],fa,fail;}tr[S];struct node{ int to,nxt;}e[S];struct data{ int x,y,id;}p[S];int m,tot,cnt,ed[S],id,head[S];int idx,pos[S],a[S],size[S],ans[S];char s[S];queue
q;inline bool cmp(data a,data b){ return a.y

+++++++++++++++++++++++++++++++++++++++++++

+本文作者:luyouqi233。               +

+欢迎访问我的博客: +

+++++++++++++++++++++++++++++++++++++++++++

转载于:https://www.cnblogs.com/luyouqi233/p/8997583.html

你可能感兴趣的文章