HDU 2896

传送门:HDU 2896

病毒侵袭

Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)
Total Submission(s): 14348    Accepted Submission(s): 3690


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
注意:
1.ASCII码可见字符的范围是32~126,如果开到0~255就会TLE。
2.题意有分歧。“病毒特征码”中的“特征”二字意味着病毒A的特征码不会是病毒B的特征码的子串。普遍认为这题的数据弱,这观点是站不住脚的,还应从题意出发。
不过代码最好还是按病毒A的特征码会是病毒B的特征码的子串来写
  1 #include<bits/stdc++.h>
  2 using namespace std;
  3 const int ls=2e2+10, lt=1e4+10, MAX_N=1e5+10, MAX_M=510;
  4 char s[ls], t[lt];
  5 int q[MAX_N], head, tail;
  6 bool used[MAX_M];
  7 struct node{
  8   int id, pre, last, pos[128];
  9   void init(){
 10     id=last=0;
 11     memset(pos, 0, sizeof(pos));
 12   }
 13 };
 14 node trie[MAX_N];
 15 void get_line(char *s){
 16   char ch;
 17   while((ch=getchar())=='\n');
 18   int i=0;
 19   s[i++]=ch;
 20   while((ch=getchar())!='\n'&&ch!=EOF) s[i++]=ch;
 21   s[i]='\0';
 22 }
 23 int build_trie(int N){
 24   int tot=0;
 25   trie[tot].init();
 26   int now;
 27   for(int id=1; id<=N; id++){
 28     get_line(s);
 29     now=0;
 30     for(int i=0; s[i]; i++){
 31       int &nt=trie[now].pos[s[i]];
 32       now=nt?nt:(trie[++tot].init(), nt=tot);
 33     }
 34     trie[now].id=id;
 35     trie[now].last=now;
 36   }
 37   return tot;
 38 }
 39 void build_ac(){
 40   head=tail=0;
 41   for(int i=0; i<128; i++){
 42     int &nt=trie[0].pos[i];
 43     if(nt){
 44       trie[nt].pre=0;
 45       q[tail++]=nt;
 46     }
 47   }
 48   int pre;
 49   while(head!=tail){
 50     int &top=q[head++];
 51     for(int i=0; i<128; i++){
 52       pre=trie[top].pre;
 53       int &nt=trie[top].pos[i];
 54       if(!nt) nt=trie[pre].pos[i];
 55       else{
 56         q[tail++]=nt;
 57         while(!trie[pre].pos[i]&&pre) pre=trie[pre].pre;
 58         pre=trie[nt].pre=trie[pre].pos[i];
 59         int &last=trie[nt].last;
 60         last=last?last:trie[pre].last;
 61       }
 62     }
 63   }
 64 }
 65 void get_ans(int last, int &cnt){
 66   while((last=trie[last].last)&&!used[trie[last].id]){  //error-prone
 67     used[trie[last].id]=true, cnt++;
 68   }
 69 }
 70 void match(int N, int V){
 71   int cnt, tot=0;
 72   for(int w=1; w<=N; w++){
 73     get_line(t);
 74     memset(used, 0, sizeof(used));
 75     cnt=0;
 76     for(int i=0, pre=0; t[i]; i++){
 77       pre=trie[pre].pos[t[i]];
 78       get_ans(pre, cnt);
 79       if(cnt>=3) break;
 80     }
 81     if(!cnt) continue;
 82     tot++;
 83     printf("web %d:", w);
 84     for(int i=1; i<=V; i++)
 85       if(used[i])
 86         printf(" %d", i);
 87     puts("");
 88   }
 89   printf("total: %d\n", tot);
 90 }
 91 int main(){
 92   freopen("in", "r", stdin);
 93   int N, M;
 94   scanf("%d", &N);
 95   build_trie(N);
 96   build_ac();
 97   scanf("%d", &M);
 98   match(M, N);
 99   return 0;
100 }

 

 
 
posted @ 2015-07-03 18:57  Pat  阅读(317)  评论(0编辑  收藏  举报