您的位置:首页 > 其它

[hdu2896][ac自动机]病毒侵袭

2018-02-06 13:25 183 查看
病毒侵袭

Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)

Total Submission(s): 32341 Accepted Submission(s): 7361

Problem Description

当太阳的光辉逐渐被月亮遮蔽,世界失去了光明,大地迎来最黑暗的时刻。。。。在这样的时刻,人们却异常兴奋——我们能在有生之年看到500年一遇的世界奇观,那是多么幸福的事儿啊~~

但网路上总有那么些网站,开始借着民众的好奇心,打着介绍日食的旗号,大肆传播病毒。小t不幸成为受害者之一。小t如此生气,他决定要把世界上所有带病毒的网站都找出来。当然,谁都知道这是不可能的。小t却执意要完成这不能的任务,他说:“子子孙孙无穷匮也!”(愚公后继有人了)。

万事开头难,小t收集了好多病毒的特征码,又收集了一批诡异网站的源码,他想知道这些网站中哪些是有病毒的,又是带了怎样的病毒呢?顺便还想知道他到底收集了多少带病毒的网站。这时候他却不知道何从下手了。所以想请大家帮帮忙。小t又是个急性子哦,所以解决问题越快越好哦~~

Input

第一行,一个整数N(1<=N<=500),表示病毒特征码的个数。

接下来N行,每行表示一个病毒特征码,特征码字符串长度在20—200之间。

每个病毒都有一个编号,依此为1—N。

不同编号的病毒特征码不会相同。

在这之后一行,有一个整数M(1<=M<=1000),表示网站数。

接下来M行,每行表示一个网站源码,源码字符串长度在7000—10000之间。

每个网站都有一个编号,依此为1—M。

以上字符串中字符都是ASCII码可见字符(不包括回车)。

Output

依次按如下格式输出按网站编号从小到大输出,带病毒的网站编号和包含病毒编号,每行一个含毒网站信息。

web 网站编号: 病毒编号 病毒编号 …

冒号后有一个空格,病毒编号按从小到大排列,两个病毒编号之间用一个空格隔开,如果一个网站包含病毒,病毒数不会超过3个。

最后一行输出统计信息,如下格式

total: 带病毒网站数

冒号后有一个空格。

Sample Input

3

aaa

bbb

ccc

2

aaabbbccc

bbaacc

Sample Output

web 1: 1 2 3

total: 1

Source

2009 Multi-University Training Contest 10 - Host by NIT

Recommend

gaojie | We have carefully selected several similar problems for you: 3065 2243 3341 2296 3746

sol:

十分naive的ac自动机,就是不要用memset,hdu是根据你访问的内存来算内存的,所以memset会mle,十分的惨惨啊,还有就是pe的问题,还有就是题目中不只有26个小写字母啊(只有我这种naive的选手才会因为这些2b问题调半天吧)

总之十分naive(之前立的flag果然炸裂了)

#include<cstdio>
#include<algorithm>
#include<string>
#include<cstring>
#include<cstdlib>
#include<cmath>
#include<iostream>
using namespace std;
int n,m;
inline int read()
{
char c;
int res,flag=0;
while((c=getchar())>'9'||c<'0') if(c=='-')flag=1;
res=c-'0';
while((c=getchar())>='0'&&c<='9') res=(res<<3)+(res<<1)+c-'0';
return flag?-res:res;
}
const int N=100002;

int tot,l,fail
,mark
;
int go
[128];
char sr[10005];/*
inline void insert(int &x,int y,int id)
{
if(!x) x=++tot;
if(y>l)
{
mark[x]=id;
return;
}
insert(go[x][sr[y]],y+1,id);
}*/
inline int newpoint()
{
++tot;
for(int i=0;i<=127;++i) go[tot][i]=0;
fail[tot]=mark[tot]=0;
return tot;
}
inline void insert(int id)
{
int v=1;
for(int i=1;i<=l;++i)
{
if(!go[v][sr[i]]) go[v][sr[i]]=newpoint();
v=go[v][sr[i]];
}
mark[v]=id;
}
int q
;
inline void build()
{
int t=0,w,u,v;
q[w=1]=1;
for(int i=0;i<=127;++i) go[0][i]=1;
while(t<w)
{
u=q[++t];
for(int ch=0;ch<=127;++ch)
if(go[u][ch])
{
q[++w]=go[u][ch];
v=fail[u];
while(!go[v][ch]) v=fail[v];
v=go[v][ch];
fail[go[u][ch]]=v;
}
}
}
int rt=1;
int Ans[5];
bool vis[510];
int main()
{
//  freopen("2896.in","r",stdin);
//  freopen("2896.out","w",stdout);
while(scanf("%d",&n)!=EOF)
{
tot=1;

for(int i=1;i<=n;++i)
{
scanf("%s",sr+1);
l=strlen(sr+1);
//          insert(rt,1,i);
insert(i);
}
build();

m=read();
int ans=0;
for(int i=1;i<=m;++i)
{
scanf("%s",sr+1);
l=strlen(sr+1);
int v=1;
Ans[0]=0;
memset(vis,0,sizeof(vis));
for(int j=1;j<=l;++j)
{
while(!go[v][sr[j]]) v=fail[v];
v=go[v][sr[j]];
int k=v;
while(k)
{
if(mark[k]&&!vis[mark[k]])
{
Ans[++Ans[0]]=mark[k];
vis[mark[k]]=1;
}
k=fail[k];
}
}
if(Ans[0])
{
printf("web %d: ",i);
sort(Ans+1,Ans+1+Ans[0]);
for(int i=1;i<Ans[0];++i)
printf("%d ",Ans[i]);
printf("%d",Ans[Ans[0]]);
printf("\n");
++ans;
}
}
printf("total: %d\n",ans);
}
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: