zoukankan      html  css  js  c++  java
  • 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())=='
    ');
     18   int i=0;
     19   s[i++]=ch;
     20   while((ch=getchar())!='
    '&&ch!=EOF) s[i++]=ch;
     21   s[i]='';
     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
    ", 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 }
     
     
  • 相关阅读:
    LocalDB数据库修改排序规则,修复汉字变问号
    设计模式读书笔记-单件模式(创建型模式)
    supersocket实现上传文件
    一步一步架起MyBatis
    windows平台下cmake+gtest工程调试
    嵌入式Linux模块移植四部曲
    一次惊呆的调试经历
    阅读《大型网站技术架构》第五章、第六章心得
    阅读《大型网站技术架构》第四章心得
    阅读《大型网站技术架构》 第三章心得
  • 原文地址:https://www.cnblogs.com/Patt/p/4619473.html
Copyright © 2011-2022 走看看