HDU 3065 病毒侵袭持续中 (AC自动机)
Problem Description
小t非常感谢大家帮忙解决了他的上一个问题。然而病毒侵袭持续中。在小t的不懈努力下,他发现了网路中的“万恶之源”。这是一个庞大的病毒网站,他有着好多好多的病毒,但是这个网站包含的病毒很奇怪,这些病毒的特征码很短,而且只包含“英文大写字符”。当然小t好想好想为民除害,但是小t从来不打没有准备的战争。知己知彼,百战不殆,小t首先要做的是知道这个病毒网站特征:包含多少不同的病毒,每种病毒出现了多少次。大家能再帮帮他吗?
Input
第一行,一个整数N(1<=N<=1000),表示病毒特征码的个数。
接下来N行,每行表示一个病毒特征码,特征码字符串长度在1—50之间,并且只包含“英文大写字符”。任意两个病毒特征码,不会完全相同。
在这之后一行,表示“万恶之源”网站源码,源码字符串长度在2000000之内。字符串中字符都是ASCII码可见字符(不包括回车)。
Output
按以下格式每行一个,输出每个病毒出现次数。未出现的病毒不需要输出。
病毒特征码: 出现次数
冒号后有一个空格,按病毒特征码的输入顺序进行输出。
Sample Input
3
AA
BB
CC
ooxxCC%dAAAoen....END
Sample Output
AA: 2
CC: 1
分析:
给你n个子串,然后从一个字符串中查找这些子串哪些出现过,出现了多少次。
代码:
#include<stdio.h>
#include<iostream>
#include<queue>
#include<string.h>
#include<algorithm>
using namespace std;
struct Node
{
Node *next[26];//病毒只涉及到26个英文字母
Node *fail;//fail指针
int id;//病毒的编号
Node()//节点初始化
{
for(int i=0; i<26; i++)
next[i]=NULL;
fail=NULL;
id=0;
}
} *tree[500009];
Node *root;
int cnt[1009];//每个病毒的出现次数
int head,tail;
char vir[1009][55];//存放病毒的信息
char source[2000009],s[2000009];
void Insert(char *str,int index)//插入信息
{
Node *p=root;
int i=0,pos;
while(str[i])
{
pos=str[i]-'A';//字母转数字
if(p->next[pos]==NULL)//如果当前的节点不存在,则建立新的节点
p->next[pos]=new Node();
p=p->next[pos];//指针接着往下指
i++;
}
p->id=index;//记录这个病毒的编号
}
void build_ac_automation(Node *root)//寻找fail指针
{
root->fail=NULL;
tree[tail++] = root;
Node *p=NULL;
while(head<tail)//相当于是一个广搜的过程
{
Node * temp=tree[head++];
for(int i=0; i<26; i++)
{
if(temp->next[i]!=NULL)//存在当前的节点
{
if(temp==root)//如果这个节点是根节点的话
temp->next[i]->fail=root;//那么他的孩子的fail指针就都是根节点
else
{
p=temp->fail;//指向当前的根节点
while(p!=NULL)
{
if(p->next[i]!=NULL)//当前结点下存在相同的节点
{
temp->next[i]->fail=p->next[i];//那么就寻找到了fail指针
break;
}
p=p->fail;
}
if(p==NULL) temp->next[i]->fail=root;//没有找到,则fail指针指向根节点
}
tree[tail++]=temp->next[i];
}
}
}
}
void query(char *str)
{
int i=0,index;
Node *p=root;
while(str[i])
{
index=str[i]-'A';//字母转数字
while(p->next[index]==NULL&&p!=root)//不是根节点,则一直寻找到fail指针
p=p->fail;
p=p->next[index];
if(p==NULL) p=root;
Node *temp=p;
while(temp!=root&&temp->id>0)//存在这个病毒
{
cnt[temp->id]++;//病毒数+
temp=temp->fail;
}
i++;
}
}
int main()
{
int n;
while(~scanf("%d",&n))
{
memset(cnt,0,sizeof(cnt));
head=tail=0;
root=new Node();
getchar();
for(int i=1; i<=n; i++)
{
scanf("%s",vir[i]);
Insert(vir[i],i);
}
build_ac_automation(root);
scanf("%s",source);
int len=strlen(source);
int l=0;
for(int i=0; i<=len; i++)
{
if(source[i]>='A'&&source[i]<='Z')
{
s[l++]=source[i];
}
else
{
s[l]='\0';
query(s);
l=0;
}
}
for(int i = 1; i <= n; i++)
{
if(cnt[i])
printf("%s: %d\n",vir[i], cnt[i]);
}
}
return 0;
}
HDU 3065 病毒侵袭持续中 (AC自动机)的更多相关文章
- hdoj 3065 病毒侵袭持续中(AC自动机)
题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=3065 思路分析:问题需要模式匹配多个模式串,需要注意的是模式串会包含和重叠,需要对AC自动机的匹配过 ...
- HDU 3065 病毒侵袭持续中
HDU 3065 病毒侵袭持续中 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) ...
- hdu 3065 病毒侵袭持续中【AC自动机】
<题目链接> 题目大意: 小t非常感谢大家帮忙解决了他的上一个问题.然而病毒侵袭持续中.在小t的不懈努力下,他发现了网路中的“万恶之源”.这是一个庞大的病毒网站,他有着好多好多的病毒,但是 ...
- HDU 3065 病毒侵袭持续中(AC自己主动机)
题目链接:http://acm.hdu.edu.cn/showproblem.php? pid=3065 Problem Description 小t非常感谢大家帮忙攻克了他的上一个问题.然而病毒侵袭 ...
- hdu3065 病毒侵袭持续中 AC自动机入门题 N(N <= 1000)个长度不大于50的模式串(保证所有的模式串都不相同), 一个长度不大于2000000的待匹配串,求模式串在待匹配串中的出现次数。
/** 题目:hdu3065 病毒侵袭持续中 链接:http://acm.hdu.edu.cn/showproblem.php?pid=3065 题意:N(N <= 1000)个长度不大于50的 ...
- hdu----(3065)病毒侵袭持续中(AC自动机)
病毒侵袭持续中 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Sub ...
- hdu 3065病毒侵袭持续中
病毒侵袭持续中 http://acm.hdu.edu.cn/showproblem.php?pid=3065 Time Limit: 2000/1000 MS (Java/Others) Mem ...
- HDU 3065 病毒侵袭持续中 (模板题)
病毒侵袭持续中 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Sub ...
- HDU3065 病毒侵袭持续中 —— AC自动机
题目链接:https://vjudge.net/problem/HDU-3065 病毒侵袭持续中 Time Limit: 2000/1000 MS (Java/Others) Memory Li ...
随机推荐
- dotTrace快速帮助你定位C#代码的性能瓶颈
dotTrace 1. 问题描述 IIS发布的接口运行一段时间后变的很慢,重启IIS连接池后问题得到解决,但是运行一段时间后再次出现变慢的问题 2. 问题原因 程序中有读取xml文件的 ...
- 爬虫学习之-requests乱码
总体功能的一个演示 import requests response = requests.get("https://www.baidu.com") print(type(resp ...
- 【php】new static的用法
在一个类中,常见的是new self()操作,代表返回自身类的实例. 当父类中存在方法,然后每个子类继承于父类,调用这个方法会返回自身的实例化对象, <?php class A { functi ...
- 2013南京网赛1003 hdu 4750 Count The Pairs
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4750 题意:给出一个无向图,f(a,b)表示从点a到点b的所有路径中的每条路径的最长边中的最小值,给出 ...
- SpringBoot(六)_AOP统一处理请求
什么是AOP AOP 是一种编程范式,与编程语言无关: 将通用逻辑从业务逻辑中分离出来(假如你的业务是一条线,我们不在业务线上写一行代码就能完成附加任务!我们会把代码写在其他的地方): 具体实现 (1 ...
- 笔记:delphi 与 Query
以下不保存证正确 Query用SQL语言执行过的,没有必要Cancel.Post,因为其会对数据库直接操作:执行Update.Insert.Delete请用SQL语句: 用Table使用对当前记录直接 ...
- P1503 鬼子进村
题目背景 小卡正在新家的客厅中看电视.电视里正在播放放了千八百次依旧重播的<亮剑>,剧中李云龙带领的独立团在一个县城遇到了一个鬼子小队,于是独立团与鬼子展开游击战. 题目描述 描述 县城里 ...
- Smallest Minimum Cut HDU - 6214(最小割集)
Smallest Minimum Cut Time Limit: 2000/2000 MS (Java/Others) Memory Limit: 65535/32768 K (Java/Oth ...
- Laravel4快速安装方法,解决Laravel4安装速度慢
Laravel4原始安装方法 Laravel4 是构建在 Composer 之上的, 之前的安装方法是如下: composer create-project laravel/laravel you ...
- 51nod乘积之和
题目链接 戳我 题意简述 你有长为\(n\)的序列和\(Q\)个询问,每次询问一个\(k\),求用\(k\)个数组成的不同方案的乘积的和. sol 显然要预处理一波. 考虑分治,左右两边都求出来后,怎 ...