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),表示病毒特征码的个数。 |
按以下格式每行一个,输出每个病毒出现次数。未出现的病毒不需要输出。 |
Sample Input - 输入样例 |
Sample Output - 输出样例 |
3 |
AA: 2 |
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 }