poj_3987 Trie图
题目大意
有N个病毒,病毒由A-Z字母构成,N个病毒各不相同。给出一段程序P,由A-Z字母构成,若病毒在在程序P或者P的逆转字符串P'中存在,则该程序P被该病毒感染。求出程序P被多少种病毒感染。
题目分析
典型的多模式串的字符串匹配问题,考虑使用Trie图。将M个待查的字符串作为模式串插入Trie图中,然后设置前缀指针,构造DFA。 
    判断程序P字符串翻转之后,是否含有某个模式串,一种方法是将P翻转,然后在DFA上查找;另一种是在构造DFA的时候,将模式串翻转,然后插入Trie图中,在匹配母串的时候就不需要将母串翻转了。 
    使用第二种方法需要注意的是,可能有两个模式串互为翻转。在Trie图的node节点中维护信息 pattern_index,若某节点为某个模式串的终止节点,则pattern_index为该模式串的序号(从1开始),若节点不是某个模式串的终止节点,则pattern_index = 0. 考虑两个模式串互为翻转(而且最多有两个模式串互为翻转)的情况,可以将pattern_index的高16bit作为pattern1的index,低16bit作为pattern2的index。
实现的时候,出现了几次超时。主要是重复访问了前缀指针节点。通过如下方法剪枝:
在trie图中遇到一个危险节点N(不一定为终止节点),此时母串遍历到当前位置P,可以确定在P之前,肯定出现了模式串
在N第一次被访问的时候,可以通过前缀指针找到N之前的所有模式串(需要不断的找prev,直到node到达根节点,比如 ABCDE中有模式串 BCDE, CDE, DE,需要不断的找前缀指针直到root,来防止遗漏某个模式串)遇到危险节点N,向前找前缀指针的时候,碰到某个之前被访问过的节点A,即可返回.这是因为:
若A为危险节点,则它肯定在第一次被访问的时候就进行和N相同的处理(向前找模式串)
若A不是危险节点,在第一次被访问的时候,通过A的前缀指针,前缀指针的前缀指针....能到达的模式串都被找到了。因此之后再次碰到A,直接返回即可。
实现(c++)
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<string.h>
#include<queue>
#include<algorithm>
using namespace std;
#define LETTERS 26
#define MAX_NODES 500000
#define MAX_VIRUS_LEN 1004
#define MAX_PROGRAM_LEN 5100005
#define MAX_VIRUS_NUM 255
char gProgram[MAX_PROGRAM_LEN];
bool gVirusVisited[MAX_VIRUS_NUM];
int gVirusFindNum;
int gVirusNum;
struct Node{
Node* childs[LETTERS];
Node* prev;
bool danger_node;
int pattern_index;
bool visited; //判断节点是否被访问过 //在trie图中遇到一个危险节点N(不一定为终止节点),此时母串遍历到当前位置P,可以确定在P之前,肯定出现了模式串
//在N第一次被访问的时候,可以通过前缀指针找到N之前的所有模式串
//(需要不断的找prev,直到node到达根节点,比如 ABCDE中有模式串 BCDE, CDE, DE,需要不断的找前缀指针直到root,来防止遗漏某个模式串) //遇到危险节点N,向前找前缀指针的时候,碰到某个之前被访问过的节点A,即可返回
//这是因为,若A为危险节点,则它肯定在第一次被访问的时候就进行和N相同的处理(向前找模式串)
//若A不是危险节点,在第一次被访问的时候,通过A的前缀指针,前缀指针的前缀指针....能到达的模式串都被找到了。因此之后
//再次碰到A,直接返回即可。
}; Node gNodes[MAX_NODES];
int gNodeCount;
void Insert(Node* root, char* str, int pat){
char*p = str;
Node* node = root;
while (*p != '\0'){
int index = *p - 'A';
if (node->childs[index] == NULL){
node->childs[index] = gNodes + gNodeCount++;
}
node = node->childs[index];
p++;
}
node->danger_node = true;
if (node->pattern_index == 0)
node->pattern_index = pat;
else{ //有可能两个virus串,互为逆串
node->pattern_index <<= 16;
node->pattern_index |= pat;
}
} void BuildDfa(){
Node* root = gNodes + 1;
for (int i = 0; i < LETTERS; i++){
gNodes[0].childs[i] = root;
}
root->prev = gNodes;
gNodes[0].prev = NULL;
queue<Node*> Q;
Q.push(root);
while (!Q.empty()){
Node* node = Q.front();
Q.pop();
Node* prev = node->prev;
Node* p;
for (int i = 0; i < LETTERS; i++){
if (node->childs[i]){
p = prev;
while (p && !p->childs[i]){
p = p->prev;
}
node->childs[i]->prev = p->childs[i];
if (p->childs[i]->danger_node)
node->childs[i]->danger_node = true;
Q.push(node->childs[i]);
}
}
}
} void FindPatternFromEndPoint(Node* node){
do{
if (node->visited) //若该节点之前被访问过,则直接返回
return; node->visited = true;
if (node->pattern_index){
if (node->pattern_index <= gVirusNum){
if (! gVirusVisited[node->pattern_index]){
gVirusVisited[node->pattern_index] = true;
gVirusFindNum++;
}
}
else{ //两个模式串互为逆串
int virus1 = node->pattern_index & 0xFFFF;
int virus2 = node->pattern_index >> 16;
if (!gVirusVisited[virus1]){
gVirusVisited[virus1] = true;
gVirusFindNum++;
}
if (!gVirusVisited[virus2]){
gVirusVisited[virus2] = true;
gVirusFindNum++;
}
}
}
node = node->prev;
} while (node->prev);
} void Search(Node* root, char* str, int n){
char*p = str;
Node* node = root;
while (*p != '\0'){
int index = *p - 'A';
if (gVirusFindNum >= n){
return;
}
while (node && node->childs[index] == NULL){
node = node->prev;
}
node = node->childs[index];
if (node->danger_node){
FindPatternFromEndPoint(node);
}
p++;
}
} int main(){
int cas;
scanf("%d", &cas);
char virus[MAX_VIRUS_LEN];
while (cas--){
int n;
memset(gNodes, 0, sizeof(gNodes));
gNodeCount = 2;
memset(gVirusVisited, false, sizeof(gVirusVisited));
gVirusFindNum = 0; scanf("%d", &n);
gVirusNum = n;
getchar();
for (int i = 0; i < n; i++){
scanf("%s", virus);
Insert(gNodes + 1, virus, i + 1);
reverse(virus, virus + strlen(virus));
Insert(gNodes + 1, virus, i + 1);
}
BuildDfa();
getchar();
char tmp;
int k = 0;
for (;;){
scanf("%c", &tmp);
if (tmp == '\n')
break; if (tmp != '['){
gProgram[k++] = tmp;
}
else{
int num;
scanf("%d", &num);
scanf("%c", &tmp);
for (int i = 0; i < num; i++){
gProgram[k++] = tmp;
}
scanf("%c", &tmp);
}
}
gProgram[k++] = '\0';
Search(gNodes + 1, gProgram, n); printf("%d\n", gVirusFindNum);
}
return 0;
}
poj_3987 Trie图的更多相关文章
- 【BZOJ-2938】病毒      Trie图 + 拓扑排序
		
2938: [Poi2000]病毒 Time Limit: 1 Sec Memory Limit: 128 MBSubmit: 609 Solved: 318[Submit][Status][Di ...
 - 【hihoCoder】1036	Trie图
		
题目:http://hihocoder.com/problemset/problem/1036 给一个词典dict,词典中包含了一些单词words.要求判断给定的一个文本串text中是否包含这个字典中 ...
 - 【hihoCoder 1036】Trie图
		
看了一下简单的$Trie图$,调模板调啊调一连调了$2h$,最后发现$-'a'$打成$-'A'$了hhh,有种摔键盘的冲动. $Trie图$是$Trie树$上建立“前缀边”,不用再像在$Trie树$上 ...
 - 字符串 --- KMP Eentend-Kmp 自动机 trie图 trie树 后缀树 后缀数组
		
涉及到字符串的问题,无外乎这样一些算法和数据结构:自动机 KMP算法 Extend-KMP 后缀树 后缀数组 trie树 trie图及其应用.当然这些都是比较高级的数据结构和算法,而这里面最常用和最熟 ...
 - Trie图和Fail树
		
Trie图和AC自动机的区别 Trie图是AC自动机的确定化形式,即把每个结点不存在字符的next指针都补全了.这样做的好处是使得构造fail指针时不需要next指针为空而需要不断回溯. 比如构造ne ...
 - hdu2457  Trie图+dp
		
hdu2457 给定n个模式串, 和一个文本串 问如果修改最少的字符串使得文本串不包含模式串, 输出最少的次数,如果不能修改成功,则输出-1 dp[i][j] 表示长度为i的字符串, 到达状态j(Tr ...
 - Trie图
		
AC自动机是KMP的多串形式,当文本串失配时,AC自动机的fail指针告诉我们应该跳到哪里去继续匹配(跳到当前匹配串的最长后缀去),所以AC自动机的状态是有限的 但是AC自动机具有不确定性, 比如要求 ...
 - CF 291E. Tree-String Problem [dfs kmp trie图优化]
		
CF291E 题意:一棵树,每条边上有一些字符,求目标串出现了多少次 直接求目标串的fail然后一边dfs一边跑kmp 然后就被特殊数据卡到\(O(n^2)\)了... 因为这样kmp复杂度分析的基础 ...
 - AC自动机相关Fail树和Trie图相关基础知识
		
装载自55242字符串AC自动机专栏 fail树 定义 把所有fail指针逆向,这样就得到了一棵树 (因为每个节点的出度都为1,所以逆向后每个节点入度为1,所以得到的是一棵树) 还账- 有了这个东西, ...
 
随机推荐
- 【WPF/C#】测试下载文件(图片)
			
需求:界面上有一个按钮,点击后联网下载一张PNG图片,保存到本地指定路径. 重要参考: http://stackoverflow.com/questions/24797485/how-to-downl ...
 - 如何测试Linux 中的wait函数能不能等待子进程的子进程?
			
#include <stdio.h> #include <stdlib.h> int main() { pid_t pid = fork(); switch(pid) { : ...
 - excel导出功能优化
			
先说说优化前,怎么做EXCEL导出功能的: 1. 先定义一个VO类,类中的字段按照EXCEL的顺序定义,并且该类只能用于EXCEL导出使用,不能随便修改. 2. 将查询到的结果集循环写入到这个VO类中 ...
 - 高性能高并发网络库:StateThreads
			
StateThreads是一个C的网络程序开发库,提供了编写高性能.高并发.高可读性的网络程序的开发库,轻量级网络应用框架 共也就3000行C代码 网络程序(Internet Application) ...
 - rails局部模板 render
			
<%= render partial: 'file' %> file是以_开头命名的文件,比如_cart.html.erb 这样就可以用render来调用了 还可以传参数 比如 rails ...
 - HTTP API 设计指南(请求部分)
			
为了保证持续和及时的更新,强烈推荐在我的Github上关注该项目,欢迎各位star/fork或者帮助翻译 前言 这篇指南介绍描述了 HTTP+JSON API 的一种设计模式,最初摘录整理自 Hero ...
 - 也谈谈js的压缩,jquery压缩。【转】
			
问题缘由: 负责公司的开发平台研发工作,考虑的知识产权的保护工作,必须要考虑java的加密技术和js脚本的加密技术.在目前java加密很容易破解的情况下,还是先搞定js的加密和压缩,一方面可以提高页面 ...
 - 获取表单提交MVC错误信息
			
if (!ModelState.IsValid) { List<string> Keys = ModelState.Ke ...
 - 关于Cocos2d-x中数据的存储提取和类型转换
			
1.获得存储在UserDefault中的变量,但是获得的变量是一个String类型的值,要用atoi函数转换为整型,但是atoi函数的传递参数是一个char*类型的值,所以用_Score.c_str( ...
 - 天线增益英文名称:antenna gain
			
天线增益是指:在输入功率相等的条件下,实际天线与理想的辐射单元在空间同一点处所产生的信号的功率密度之比.它定量地描述一个天线把输入功率集中辐射的程度.增益显然与天线方向图有密切的关系,方向图主瓣越窄, ...