【刷题】洛谷 P3796 【模板】AC自动机(加强版)

题目描述

\(N\) 个由小写字母组成的模式串以及一个文本串 \(T\) 。每个模式串可能会在文本串中出现多次。你需要找出哪些模式串在文本串 \(T\) 中出现的次数最多。

输入输出格式

输入格式:

输入含多组数据。

每组数据的第一行为一个正整数 \(N\) ,表示共有 \(N\) 个模式串, \(1 \leq N \leq 150\)

接下去 \(N\) 行,每行一个长度小于等于 \(70\) 的模式串。下一行是一个长度小于等于 \(10^6\) 的文本串 \(T\)

输入结束标志为 \(N=0\)

输出格式:

对于每组数据,第一行输出模式串最多出现的次数,接下去若干行每行输出一个出现次数最多的模式串,按输入顺序排列。

输入输出样例

输入样例#1:

2
aba
bab
ababababac
6
beta
alpha
haha
delta
dede
tata
dedeltalphahahahototatalpha
0

输出样例#1:

4
aba
2
alpha
haha

题解

同样的,简单匹配,匹配到有结尾标记的点,就把标记的点的贡献加上

最后取个最大值,找串输出就好了

因为可能存在相同的串,结尾标记用vector存就好了

#include<bits/stdc++.h>
#define ui unsigned int
#define ll long long
#define db double
#define ld long double
#define ull unsigned long long
const int MAXN=150+10,MAXS=1000000+10,MAXM=15000+10;
int n,ch[MAXM][30],fail[MAXM],last[MAXM],ed[MAXM],ans[MAXN],cnt;
char s[MAXS],t[MAXN][80];
std::queue<int> q;
std::vector<int> V[MAXM];
template<typename T> inline void read(T &x)
{
	T data=0,w=1;
	char ch=0;
	while(ch!='-'&&(ch<'0'||ch>'9'))ch=getchar();
	if(ch=='-')w=-1,ch=getchar();
	while(ch>='0'&&ch<='9')data=((T)data<<3)+((T)data<<1)+(ch^'0'),ch=getchar();
	x=data*w;
}
template<typename T> inline void write(T x,char ch='\0')
{
	if(x<0)putchar('-'),x=-x;
	if(x>9)write(x/10);
	putchar(x%10+'0');
	if(ch!='\0')putchar(ch);
}
template<typename T> inline void chkmin(T &x,T y){x=(y<x?y:x);}
template<typename T> inline void chkmax(T &x,T y){x=(y>x?y:x);}
template<typename T> inline T min(T x,T y){return x<y?x:y;}
template<typename T> inline T max(T x,T y){return x>y?x:y;}
inline void init()
{
	for(register int i=0,lt=strlen(s);i<lt;++i)s[i]-='a'-1;
}
inline void insert(int rk)
{
	int x=0;
	for(register int i=0,lt=strlen(s);i<lt;++i)
		if(!ch[x][s[i]])x=ch[x][s[i]]=++cnt;
		else x=ch[x][s[i]];
	ed[x]=1;
	V[x].push_back(rk);
}
inline void getfail()
{
	for(register int i=1;i<=26;++i)
		if(ch[0][i])q.push(ch[0][i]);
	while(!q.empty())
	{
		int x=q.front();
		q.pop();
		for(register int i=1,y,z;i<=26;++i)
			if(ch[x][i])
			{
				y=ch[x][i],z=fail[x];
				while(z&&!ch[z][i])z=fail[z];
				fail[y]=ch[z][i];
				last[y]=ed[fail[y]]?fail[y]:last[fail[y]];
				q.push(y);
			}
			else ch[x][i]=ch[fail[x]][i];
	}
}
inline void save(int x)
{
	if(x)
	{
		if(ed[x])
			for(register int i=0,lt=V[x].size();i<lt;++i)ans[V[x][i]]++;
		save(last[x]);
	}
}
inline void match()
{
	int x=0;
	for(register int i=0,lt=strlen(s);i<lt;++i)x=ch[x][s[i]],save(ed[x]||last[x]?x:0);
}
int main()
{
	while(scanf("%d",&n)!=EOF&&n)
	{
		for(register int i=1;i<=cnt;++i)V[i].clear();
		memset(ch,0,sizeof(ch));
		memset(ed,0,sizeof(ed));
		memset(ans,0,sizeof(ans));
		memset(fail,0,sizeof(fail));
		memset(last,0,sizeof(last));
		cnt=0;
		for(register int i=1;i<=n;++i)
		{
			scanf("%s",s);
			memcpy(t[i],s,sizeof(t[i]));
			init();insert(i);
		}
		getfail();
		scanf("%s",s);init();match();
		int len=0;
		for(register int i=1;i<=n;++i)chkmax(len,ans[i]);
		write(len,'\n');
		for(register int i=1;i<=n;++i)
			if(ans[i]==len)puts(t[i]);
	}
	return 0;
}
posted @ 2018-07-12 21:30  HYJ_cnyali  阅读(196)  评论(0编辑  收藏  举报