[BZOJ2434]NOI2011阿狸的打字机|AC自动机|fail树|树状数组

本文介绍了一种使用AC自动机和Fail树解决字符串匹配问题的方法,通过优化的插入和查询策略,实现高效的字符串匹配搜索。文章详细阐述了如何构建自动机并利用Fail树进行快速查询。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

这题真是太神了,好多实用的技巧。。首先肯定是要先把每个要输出的串当模式串把自动机给建出来的,如果一个一个串复制出来再一个个插入显然非常慢。。。我们用在自动机上插入模式串的方法来建,初始时在0,新加一个字符就想下爬(或者新建),维护一个父亲指针,删除的时候就可以爬上去,这样就可以O(n)建出来了。。

再考虑询问的问题,每次把串拿出来再放进自动机跑一遍显然太慢。。这里需要用到一个叫做fail树的东西,就是把fail指针当做边建成的一颗树。。比如fail(i)=j,那么i在fail树上的父节点就是j。。对于每个询问,若x串在y串中出现了,那么root到y串节点的路径上必有一点的next指针是x串节点,放到fail树上看,询问(x,y)的答案就是x在fail上的子树中在root到y串节点的路径上的节点的数量,看到子树和,考虑用dfs序来解决。。把所有询问像链表一样挂到各自的y串节点上,然后模拟一开始的插入过程,每向下爬到一个节点就将它在dfs序中的左边界+1,每向上离开一个节点就将它在dfs序中的左边界-1,每当爬到一个danger节点就把所有这个节点的询问解决,用树状数组维护前缀和即可。。

#include<iostream>
#include<cstdio>
#include<memory.h>
#define N 100005
using namespace std;
char s[N];
struct edge{
	int e,next;
}ed[N];
struct Query{
	int e,next,xu;
}q[N];
int i,j,ne=0,x,y,t=0,nq=0,n,m,a[N][27],head[N],que[N],next[N],fa[N],l[N],r[N],c[N*2],g[N],ans[N],pos[N];//cnt为串数,nd为AC自动机点数,ne为fail树边数,nq为询问链表边数 
void inse(int s,int e)
{
	ed[++ne].e=e;
	ed[ne].next=g[s];g[s]=ne;
}
void insq(int x,int y,int i)
{
	q[++nq].e=x;q[nq].xu=i;
	q[nq].next=head[y];head[y]=nq;
}
int lowbit(int x){return x&-x;}
void add(int x,int k)
{
	for (int i=x;i<=t;i+=lowbit(i)) c[i]+=k;
}
int sum(int x)
{
	int ans=0;
	for (int i=x;i;i-=lowbit(i)) ans+=c[i];
	return ans;
}
void init()
{
	int i,now=1,cnt=0,nd=1;
	memset(a,0,sizeof(a));memset(c,0,sizeof(c));
	memset(head,0,sizeof(head));memset(g,0,sizeof(g));
	n=strlen(s);fa[1]=0;
	for (i=0;i<26;i++) a[0][i]=1;
	for (i=0;i<n;i++)
		if (s[i]=='P') pos[++cnt]=now;
		else if (s[i]=='B') now=fa[now];
		else
		{
			if (!a[now][s[i]-'a'])
			{
				a[now][s[i]-'a']=++nd;
				fa[nd]=now;
			}
			now=a[now][s[i]-'a'];
		}
}
void build()
{
	next[1]=0;inse(0,1);
	int head=1,tail=1,get,k,i;
	que[1]=1;
	while (head<=tail)
	{
		get=que[head++];
		for (i=0;i<26;i++)
			if (a[get][i])
			{
				k=next[get];
				while (!a[k][i]) k=next[k];k=a[k][i];
				next[a[get][i]]=k;inse(k,a[get][i]);
				que[++tail]=a[get][i];
			}
	}
}
void dfs(int x)
{
	l[x]=++t;
	for (int j=g[x];j;j=ed[j].next)
		dfs(ed[j].e);
	r[x]=++t;
}
void solve()
{
	int now=1,cnt=0,i,j;
	add(l[1],1);
	for (i=0;i<n;i++)
		if (s[i]=='P')
		{
			cnt++;
			for (j=head[now];j;j=q[j].next)
				ans[q[j].xu]=sum(r[q[j].e])-sum(l[q[j].e]-1);
		}
		else if (s[i]=='B') add(l[now],-1),now=fa[now];
		else now=a[now][s[i]-'a'],add(l[now],1);
}
int main()
{
	freopen("2434.in","r",stdin);
	scanf("%s",s);
	scanf("%d",&m);
	init();build();dfs(0);
	for (i=1;i<=m;i++)
	{
		scanf("%d%d",&x,&y);
		insq(pos[x],pos[y],i);
	}
	solve();
	for (i=1;i<=m;i++) printf("%d\n",ans[i]);
}


资源下载链接为: https://pan.quark.cn/s/9648a1f24758 这个HTML文件是一个专门设计的网页,适合在告白或纪念日这样的特殊时刻送给女朋友,给她带来惊喜。它通过HTML技术,将普通文字转化为富有情感和创意的表达方式,让数字媒体也能传递深情。HTML(HyperText Markup Language)是构建网页的基础语言,通过标签描述网页结构和内容,让浏览器正确展示页面。在这个特效网页中,开发者可能使用了HTML5的新特性,比如音频、视频、Canvas画布或WebGL图形,来提升视觉效果和交互体验。 原本这个文件可能是基于ASP.NET技术构建的,其扩展名是“.aspx”。ASP.NET是微软开发的一个服务器端Web应用程序框架,支持多种编程语言(如C#或VB.NET)来编写动态网页。但为了在本地直接运行,不依赖服务器,开发者将其转换为纯静态的HTML格式,只需浏览器即可打开查看。 在使用这个HTML特效页时,建议使用Internet Explorer(IE)浏览器,因为一些老的或特定的网页特效可能只在IE上表现正常,尤其是那些依赖ActiveX控件或IE特有功能的页面。不过,由于IE逐渐被淘汰,现代网页可能不再对其进行优化,因此在其他现代浏览器上运行可能会出现问题。 压缩包内的文件“yangyisen0713-7561403-biaobai(html版本)_1598430618”是经过压缩的HTML文件,可能包含图片、CSS样式表和JavaScript脚本等资源。用户需要先解压,然后在浏览器中打开HTML文件,就能看到预设的告白或纪念日特效。 这个项目展示了HTML作为动态和互动内容载体的强大能力,也提醒我们,尽管技术在进步,但有时复古的方式(如使用IE浏览器)仍能唤起怀旧之情。在准备类似的个性化礼物时,掌握基本的HTML和网页制作技巧非常
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值