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又是个急性子哦,所以解决问题越快越好哦~~
但 网路上总有那么些网站,开始借着民众的好奇心,打着介绍日食的旗号,大肆传播病毒。小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码可见字符(不包括回车)。
接下来N行,每行表示一个病毒特征码,特征码字符串长度在20—200之间。
每个病毒都有一个编号,依此为1—N。
不同编号的病毒特征码不会相同。
在这之后一行,有一个整数M(1<=M<=1000),表示网站数。
接下来M行,每行表示一个网站源码,源码字符串长度在7000—10000之间。
每个网站都有一个编号,依此为1—M。
以上字符串中字符都是ASCII码可见字符(不包括回车)。
Output
依次按如下格式输出按网站编号从小到大输出,带病毒的网站编号和包含病毒编号,每行一个含毒网站信息。
web 网站编号: 病毒编号 病毒编号 …
冒号后有一个空格,病毒编号按从小到大排列,两个病毒编号之间用一个空格隔开,如果一个网站包含病毒,病毒数不会超过3个。
最后一行输出统计信息,如下格式
total: 带病毒网站数
冒号后有一个空格。
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 }