zoukankan      html  css  js  c++  java
  • HDU 3065 病毒侵袭持续中

    HDU 3065 病毒侵袭持续中

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

    Problem Description - 题目描述

    t非常感谢大家帮忙解决了他的上一个问题。然而病毒侵袭持续中。在小t的不懈努力下,他发现了网路中的万恶之源。这是一个庞大的病毒网站,他有着好多好多的病毒,但是这个网站包含的病毒很奇怪,这些病毒的特征码很短,而且只包含英文大写字符。当然小t好想好想为民除害,但是小t从来不打没有准备的战争。知己知彼,百战不殆,小t首先要做的是知道这个病毒网站特征:包含多少不同的病毒,每种病毒出现了多少次。大家能再帮帮他吗?

     

    Input - 输入

    Output - 输出

    第一行,一个整数N(1<=N<=500),表示病毒特征码的个数。
    接下来N行,每行表示一个病毒特征码,特征码字符串长度在20—200之间。
    每个病毒都有一个编号,依此为1—N。
    不同编号的病毒特征码不会相同。
    在这之后一行,有一个整数M(1<=M<=1000),表示网站数。
    接下来M行,每行表示一个网站源码,源码字符串长度在7000—10000之间。
    每个网站都有一个编号,依此为1—M。
    以上字符串中字符都是ASCII码可见字符(不包括回车)。

    按以下格式每行一个,输出每个病毒出现次数。未出现的病毒不需要输出。
    病毒特征码: 出现次数
    冒号后有一个空格,按病毒特征码的输入顺序进行输出。

    Sample Input - 输入样例

    Sample Output - 输出样例

    3
    AA
    BB
    CC
    ooxxCC%dAAAoen....END

    AA: 2
    CC: 1

     

    Hint - 提示

    题目描述中没有被提及的所有情况都应该进行考虑。比如两个病毒特征码可能有相互包含或者有重叠的特征码段。

    计数策略也可一定程度上从Sample中推测。

    【题解】
      每次匹配的时候不清除即可,一般的AC自动机,坑点是多组输入,以及眼神不好……

    【代码 C++】

     1 #include <cstdio>
     2 #include <cstring>
     3 #include <algorithm>
     4 #include <queue>
     5 int tr[50005][26], iTR, fail[50005], sWord[1005], cnt[50005];
     6 char  word[1005][55], text[2000005];
     7 int n;
     8 void build(){
     9     memset(tr, 0, sizeof(tr)); memset(fail, 0, sizeof(fail));
    10     memset(cnt, 0, sizeof(cnt)); memset(sWord, 0, sizeof(sWord));
    11     iTR = 0;
    12     int i, j, k, w;
    13 
    14     for (i = 1; i <= n; ++i){
    15         gets(word[i]);
    16         for (j = k = 0; word[i][j]; ++j){
    17             w = word[i][j] - 'A';
    18             if (!tr[k][w]) tr[k][w] = ++iTR;
    19             k = tr[k][w];
    20         }
    21         cnt[k] = i;
    22     }
    23 }
    24 void setFail(){
    25     std::queue<int> q;
    26     int i, j, now;
    27     for (i = 0; i < 26; ++i) if (tr[0][i]) q.push(tr[0][i]);
    28     while (!q.empty()){
    29         now = q.front(); q.pop();
    30         for (i = 0; i < 26; ++i){
    31             if (j = tr[now][i]) q.push(j), fail[j] = tr[fail[now]][i];
    32             else tr[now][i] = tr[fail[now]][i];
    33         }
    34     }
    35 }
    36 void fid(){
    37     gets(text);
    38     int i, j, temp;
    39     for (i = j = 0; text[i]; ++i){
    40         if (text[i] < 'A' || 'Z' < text[i]){ j = 0; continue; }
    41         j = tr[j][text[i] - 'A'];
    42         for (temp = j; temp; temp = fail[temp]) ++sWord[cnt[temp]];
    43     }
    44 }
    45 void opt(){
    46     for (int i = 1; i <= n; ++i){
    47         if (sWord[i]) printf("%s: %d
    ", word[i], sWord[i]);
    48     }
    49 }
    50 int main(){
    51     while (~scanf("%d", &n)){
    52         getchar();
    53         build(); setFail(); fid();
    54         opt();
    55     }
    56     return 0;
    57 }


     

  • 相关阅读:
    When You Get Troubles
    CentOS 6.8升级到7+
    Tomcat服务器搭建
    Percona Server 安装
    VirtualBox中如何使虚拟机能够上网?
    CentOS
    xen安装
    SSH免密码设置
    打造绿色版的RobotFramework
    零散知识记录-Jira的安装
  • 原文地址:https://www.cnblogs.com/Simon-X/p/5687318.html
Copyright © 2011-2022 走看看