zoj 3430 Detect the Virus(AC自己主动机)
题目连接:zoj 3430 Detect the Virus
题目大意:给定一个编码完的串,将每个字符相应着表的数值转换成6位二进制。然后以8为一个数值,又一次形成字符
串,推断给定询问串是否含有字符集中的串。
解题思路:主要是题意,逆编码部分注意。转换完了之后,可能有字符'\0'。所以不能用字符串的形式储存。要用int型
的数组。
注意有同样串的可能。
#include <cstdio>
#include <cstring>
#include <queue>
#include <vector>
#include <iostream>
#include <algorithm>
using namespace std;
typedef long long ll;
const int maxn = 512 * 100;
const int sigma_size = 256;
struct Aho_Corasick {
int sz, g[maxn][sigma_size];
int tag[maxn], fail[maxn], last[maxn];
bool vis[520];
int jump[520];
void init();
int idx(char ch);
void insert(int* str, int k);
void getFail();
int match(int* str);
void put(int u);
}A;
inline int change(char ch) {
if (ch >= 'A' && ch <= 'Z')
return ch - 'A';
if (ch >= 'a' && ch <= 'z')
return ch - 'a' + 26;
if (ch >= '0' && ch <= '9')
return ch - '0' + 52;
if (ch == '+')
return 62;
return 63;
}
const int maxl = 5005;
char w[maxl];
int s[maxl * 2];
void input(int* s) {
scanf("%s", w);
s[0]=0;
int n = strlen(w);
while (n && w[n-1] == '=') n--;
w[n] = 0;
for(int i=0,len=0,x=0;w[i];++i){
len+=6,x=(x<<6)|change(w[i]);
if(len>=8){
s[++s[0]]=(x>>(len-8))&0xff;
len-=8;
}
}
/*
for(int i = 1; i <= s[0]; i++)
printf(" %d", s[i]);
printf("\n");
*/
}
int N, M;
int main () {
while (scanf("%d", &N) == 1) {
A.init();
for (int i = 1; i <= N; i++) {
input(s);
A.insert(s, i);
}
A.getFail();
scanf("%d", &M);
for (int i = 1; i <= M; i++) {
input(s);
printf("%d\n", A.match(s));
}
printf("\n");
}
return 0;
}
void Aho_Corasick::init() {
sz = 1;
tag[0] = 0;
memset(g[0], 0, sizeof(g[0]));
}
int Aho_Corasick::idx(char ch) {
return ch;
}
void Aho_Corasick::put(int u) {
int p = tag[u];
while (p && vis[p] == 0) {
vis[p] = 1;
p = jump[p];
}
if (last[u])
put(last[u]);
}
void Aho_Corasick::insert(int* str, int k) {
int u = 0;
for (int i = 1; i <= str[0]; i++) {
int v = str[i];
while (v >= sigma_size);
if (g[u][v] == 0) {
tag[sz] = 0;
memset(g[sz], 0, sizeof(g[sz]));
g[u][v] = sz++;
}
u = g[u][v];
}
jump[k] = tag[u];
tag[u] = k;
}
int Aho_Corasick::match(int* str) {
memset(vis, 0, sizeof(vis));
int u = 0;
for (int i = 1; i <= str[0]; i++) {
int v = str[i];
while (v >= sigma_size);
while (u && g[u][v] == 0)
u = fail[u];
u = g[u][v];
if (tag[u])
put(u);
else if (last[u])
put(last[u]);
}
int ret = 0;
for (int i = 1; i <= N; i++)
if (vis[i])
ret++;
return ret;
}
void Aho_Corasick::getFail() {
queue<int> que;
for (int i = 0; i < sigma_size; i++) {
int u = g[0][i];
if (u) {
fail[u] = last[u] = 0;
que.push(u);
}
}
while (!que.empty()) {
int r = que.front();
que.pop();
for (int i = 0; i < sigma_size; i++) {
int u = g[r][i];
if (u == 0) {
g[r][i] = g[fail[r]][i];
continue;
}
que.push(u);
int v = fail[r];
while (v && g[v][i] == 0)
v = fail[v];
fail[u] = g[v][i];
last[u] = tag[fail[u]] ?
fail[u] : last[fail[u]];
}
}
}
zoj 3430 Detect the Virus(AC自己主动机)的更多相关文章
- ZOJ - 3430 Detect the Virus —— AC自动机、解码
题目链接:https://vjudge.net/problem/ZOJ-3430 Detect the Virus Time Limit: 2 Seconds Memory Limit: 6 ...
- zoj 3430 Detect the Virus(AC自己主动机)
Detect the Virus Time Limit: 2 Seconds Memory Limit: 65536 KB One day, Nobita found that his co ...
- ZOJ 3430 Detect the Virus
传送门: Detect the Virus ...
- ZOJ - 3228 Searching the String (AC自己主动机)
Description Little jay really hates to deal with string. But moondy likes it very much, and she's so ...
- ZOJ 3494 BCD Code (AC自己主动机 + 数位DP)
题目链接:BCD Code 解析:n个病毒串.问给定区间上有多少个转换成BCD码后不包括病毒串的数. 很奇妙的题目. . 经典的 AC自己主动机 + 数位DP 的题目. 首先使用AC自己主动机,得到b ...
- ZOJ 3430 Detect the Virus(AC自动机)
题目链接:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3430 题意:给你n个编码后的模式串,和m个编码后的主串,求原来主 ...
- ZOJ 3430 Detect the Virus(AC自动机 + 模拟)题解
题意:问你主串有几种模式串.但是所有串都是加密的,先解码.解码过程为:先把串按照他给的映射表变成6位数二进制数,然后首尾衔接变成二进制长串,再8位8位取变成新的数,不够的补0.因为最多可能到255,所 ...
- ZOJ 3430 Detect the Virus 【AC自动机+解码】
解码的那些事儿,不多说. 注意解码后的结果各种情况都有,用整数数组存储,char数组会超char类型的范围(这个事最蛋疼的啊)建立自动机的时候不能用0来判断结束. #include <cstdi ...
- Zoj 3545 Rescue the Rabbit(ac自己主动机+dp)
标题效果: 鉴于DNA有一个正确的顺序值.请构造一个长度I的DNA在这个序列使DNA正确的顺序值极大.它被认为是负的输出噼啪. .. IDEAS: 施工顺序是,ac己主动机上走,求最大要用到dp dp ...
随机推荐
- [codeforces438E]The Child and Binary Tree
[codeforces438E]The Child and Binary Tree 试题描述 Our child likes computer science very much, especiall ...
- RocketMQ 源码分析 RouteInfoManager(四)
在上一章分析了NamesrvController的构造函数时,会生成一个RouteInfoManager对象,该对象存放着整个消息集群的相关消息,所以这里单独拿出来分析.其实试想一下namesrv的功 ...
- transform perspective的层级问题
如上图,在积分的数字元素上,使用了transform perspective,其层级就穿透了上面的遮罩层,关键代码如下: .mask { position: fixed; z-index:; } .f ...
- BFC,IFC,GFC,FFC
FC的全称是:Formatting Contexts,是W3C CSS2.1规范中的一个概念.它是页面中的一块渲染区域,并且有一套渲染规则,它决定了其子元素将如何定位,以及和其他元素的关系和相互作用. ...
- 【转】IDEA 2017破解 license server激活
确保电脑在联网状态,在激活窗口选择license server 填入下面的license server: http://intellij.mandroid.cn/ http://idea.imsxm. ...
- 《c程序设计语言》读书笔记-5.4-指针实现strend
#include <stdio.h> #include <math.h> #include <stdlib.h> #include <string.h> ...
- 切换cmd的目录
http://jingyan.baidu.com/article/af9f5a2d20253343140a450f.html
- clips 前端 js 倒计时 获取验证码的按钮
<a href="javascript:void(0);" onclick="get_captcha()" class="btn btn-def ...
- bzoj4897 [Thu Summer Camp2016]成绩单
传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=4897 [题解] 第一次看这题想的是f[l,r]的区间dp发现仅记录这两个好像不能转移啊 会出 ...
- ws2s函数
std::string ws2s(const std::wstring& str) { char* pElementText; int iTextLen; // wide char to mu ...