P2414 [NOI2011]阿狸的打字机
AC自动机+树状数组
优质题解 <------题目分析
先AC自动机搞出Trie图
然后根据fail指针建一只新树
把树映射(拍扁)到一个序列上,用树状数组加速优化
在新树上处理时间戳,用于树状数组维护
在原Trie树上跑dfs查询答案。
因为Trie下标从0开始,树状数组从1开始
所以所有有关的下标都要注意
attention:树状数组上限一定要+1!(就是它卡了我1天TAT)
#include<iostream>
#include<cstdio>
#include<cstring>
#include<queue>
using namespace std;
inline int max(int &a,int &b) {return a>b ?a:b;}
const int N=1e5+;
struct Trie{int nxt_[],nxt[],fail,end,fa;}a[N];
int n,tot,cnt,word[N],ans[N];
struct AC_automaton{ //AC自动机
char q[N];
void Trie_build(){
scanf("%s",q);
int u=,len=strlen(q),st=;
while(q[st]=='B'||q[st]=='P') ++st;
for(int i=st;i<len;++i){
if(q[i]=='B') u=a[u].fa; //删除一位
else if(q[i]=='P') word[++tot]=u,a[u].end=tot; //加入新单词
else{
int p=q[i]-'a';
if(!a[u].nxt[p]) a[u].nxt[p]=++cnt,a[cnt].fa=u;
u=a[u].nxt[p];
}
}
}
void AC_build(){
queue <int> h;
for(int i=;i<;++i) if(a[].nxt[i]) h.push(a[].nxt[i]);
while(!h.empty()){
int x=h.front(); h.pop();
for(int i=;i<;++i){
int &to=a[x].nxt[i];
if(to){
a[to].fail=a[a[x].fail].nxt[i];
h.push(to);
}else to=a[a[x].fail].nxt[i];
}
}
}
void backup(){ //对Trie树的nxt进行备份由于下面的遍历(AC自动机会改变nxt)
for(int i=;i<=cnt;++i)
for(int j=;j<;++j)
a[i].nxt_[j]=a[i].nxt[j];
}
}mo1;
struct tree_array{ //树状数组
int c[N];
inline void add(int x,int k) {for(;x<=cnt+;x+=x&-x) c[x]+=k;} //上限要+1!
inline int sum(int x){int res=; for(;x;x-=x&-x) res+=c[x]; return res;}
}mo2;
int cnt1,hd1[N],nxt1[N],ed1[N],poi1[N];
int cnt2,hd2[N],nxt2[N],ed2[N],poi2[N],id[N];
inline void add_edge1(int x,int y){ //fail树邻接表
nxt1[ed1[x]]=++cnt1; hd1[x]= hd1[x] ? hd1[x]:cnt1;
ed1[x]=cnt1; poi1[cnt1]=y;
}
inline void add_edge2(int x,int y,int _id){ //存询问邻接表
nxt2[ed2[x]]=++cnt2; hd2[x]= hd2[x] ? hd2[x]:cnt2;
ed2[x]=cnt2; poi2[cnt2]=y; id[cnt2]=_id;
}
struct new_tree{
int dfs_clock,dfn[N],low[N];
inline void dfs1(int x){ //fail树遍历
dfn[x]=++dfs_clock; //时间戳
for(int i=hd1[x];i;i=nxt1[i]) dfs1(poi1[i]);
low[x]=dfs_clock; //size[x]=low[x]-dfn[x]
}
inline void dfs2(int x){ //Trie树遍历
mo2.add(dfn[x],); //保证只有该条路径上
if(a[x].end) //该点是某个单词的结尾
{
for(int i=hd2[a[x].end];i;i=nxt2[i]){
int e=word[poi2[i]];
ans[id[i]]=mo2.sum(low[e])-mo2.sum(dfn[e]-);
}
}
for(int i=;i<;++i) if(a[x].nxt_[i]) dfs2(a[x].nxt_[i]); //沿原树遍历
mo2.add(dfn[x],-);
}
}mo3;
int main(){
mo1.Trie_build(); mo1.backup(); mo1.AC_build();
for(int i=;i<=cnt;++i) add_edge1(a[i].fail,i); //fail树连边
scanf("%d",&n); int q1,q2;
for(int i=;i<=n;++i) scanf("%d%d",&q1,&q2),add_edge2(q2,q1,i); //把询问存到邻接表上
mo3.dfs1(); mo3.dfs2();
for(int i=;i<=n;++i) printf("%d\n",ans[i]);
return ;
}
P2414 [NOI2011]阿狸的打字机的更多相关文章
- 洛谷 P2414 [NOI2011]阿狸的打字机 解题报告
P2414 [NOI2011]阿狸的打字机 题目背景 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机. 题目描述 打字机上只有28个按键,分别印有26个小写英文字母和'B'.'P'两个字母 ...
- 洛谷P2414 - [NOI2011]阿狸的打字机
Portal Description 首先给出一个只包含小写字母和'B'.'P'的操作序列\(s_0(|s_0|\leq10^5)\).初始时我们有一个空串\(t\),依次按\(s_0\)的每一位进行 ...
- 【AC自动机】【树状数组】【dfs序】洛谷 P2414 [NOI2011]阿狸的打字机 题解
这一题是对AC自动机的充分理解和树dfs序的巧妙运用. 题目背景 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机. 题目描述 打字机上只有28个按键,分别印有26个小写英文字母和' ...
- 洛谷P2414 [NOI2011]阿狸的打字机(AC自动机)
传送门 考虑一下,如果串B在串A中出现过,那么A的fail指针必定直接或间接指向B 那么我们可以把fail树建起来,那么就变成B代表的节点的子树里有多少节点属于A 然后这就是一个序列统计问题,直接用d ...
- BZOJ 2434 Luogu P2414 [NOI2011]阿狸的打字机 (AC自动机、树状数组)
题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=2434 题解: 我写的是离线做法,不知道有没有在线做法. 转化一波题意,\(x\)在AC ...
- P2414 [NOI2011]阿狸的打字机 AC自动机
题意 给定n个模式串,有m个询问,每次询问第X个模式串在第Y个模中出现了多少次 解题思路 以fail树相反的方向建一棵树T,问题转化为X的子树中有多少个y的终止节点.跑出T的dfs序,X的子树就可以表 ...
- BZOJ2434: [NOI2011]阿狸的打字机(AC自动机+dfs序+树状数组)
[NOI2011]阿狸的打字机 题目链接:https://www.luogu.org/problemnew/show/P2414 题目背景 阿狸喜欢收藏各种稀奇古怪的东西,最近他淘到一台老式的打字机. ...
- 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序+树状 ...
随机推荐
- @font-face 字体图标的应用
所谓字体图标,顾名思义就是图标以字体的形式存在,可以利用 font-size.color 对字体图标的大小和颜色进行渲染.将小图标集中放到字体库里,利用css3 @font-face 引用图标,不仅有 ...
- python requests模块的两个方法content和text
requests模块下有两个获取内容的方法,很奇怪,都是获取请求后内容的方法,有什么区别呢?? 一.区别 content:返回bytes类型的数据也就是二进制数据 text:返回的就是纯文本(Unic ...
- win7 链接打印机时提示未知的用户名或错误密码
使用win7系统时,访问局域网内的计算机 \\ip 要求输入正确用户名和密码,输入用户名和密码后,还是一直提示“未知的用户名或错误密码”. 解决方法: 开始---->运行 打开组策略编辑器 gp ...
- 常用社交网络(微博等)动态新闻(feed、新鲜事)系统浅析(转)
add by zhj:同时也看看国外牛逼公司是怎么做的吧 Stream-Framework Python实现的feed Twitter 2013 Redis based, database fa ...
- mybatis之入门
一.mybatis介绍 是apache旗下的一个开源的顶级ORM框架(做dao层的操作) 开始叫ibatis在2010年经过升级后发布到google code上就改名为mybatis 定位:1.是一个 ...
- sql server字符串中怎么添加换行?
换行/回车,可以使用CHAR函数处理,比如: 1 insert into tbtest (text) values ('abc' + char(13)+char(10) + 'def') 主要还是要看 ...
- laravel教程入门笔记
安装laravel框架 1.安装命令 composer create-project --prefer-dist laravel/laravel ytkah ytkah表示文件夹名,如果不写的话自动会 ...
- backreference Oracle正則表達式中的反向引用
这是Oracle对正則表達式的backreference的描写叙述 从定义中能够看到,当匹配表达式中已()的形式将一个子串包括起来.后面就能够以\? 的形式来引用.\1相应第一个(),\2相应第二 ...
- 8 jmeter之集合点
集合点:集合点用以同步虚拟用户,以便恰好在同一时刻执行任务.在测试计划中,可能会要求系统能够承受1000 人同时提交数据,在LoadRunner 中可以通过在提交数据操作前面加入集合点,这样当虚拟用户 ...
- PHP 常用命令行
1.PHP运行指定文件 php -f test.php (-f 可省略) 2.命令行直接运行PHP代码 php -r "phpinfo();" 如果结果太长,还可以 php -r ...