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 ...
随机推荐
- JS计算器(自制)
<!doctype html><html><header><meta charset="utf-8"><script src= ...
- 个人收藏的移动端网页布局rem解决方案
写移动端项目时,总是会纠结是用css3 media query 还是用rem.移动端框架挺多,但是因为项目都比较小,不考虑使用. 无意在网上找到一个移动端rem布局的解决方案,经个人实践,目前未出现什 ...
- BZOJ3236 [Ahoi2013]作业 【莫队 + 树状数组】
题目链接 BZOJ3236 题解 没想到这题真的是如此暴力 #include<algorithm> #include<iostream> #include<cstring ...
- 雅礼集训 Day7 T1 Equation 解题报告
Reverse 题目背景 小\(\text{G}\)有一个长度为\(n\)的\(01\)串\(T\),其中只有\(T_S=1\),其余位置都是\(0\).现在小\(\text{G}\)可以进行若干次以 ...
- 更改Debian的console分辨率
1.通过apt-get install hwinfo来安装 2.通过命令hwinfo --framebuffer来获取 eric@debian:~$ sudo sudo hwinfo --frameb ...
- css垂直居中的几种方式
1. 对于可以一行处理的 设置 height:apx; line-height:apx; 2.对于一段文字(会多行显示的) ->2.1如果是可以设置一个固定高度的 ...
- Java Nio注意事项
Selector : public abstract class Selector extends Object SelectableChannel 对象的多路复用器. 可通过调用此类的 open ...
- javascript简单拖拽(鼠标事件 mousedown mousemove mouseup)
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xht ...
- mysql 取消命令行继续编辑
mysql> create database mingongge defa\c#回车 置空mysql> 加一个\c cancel 编辑命令 回车
- JAVA版拆分大整数为2幂的和算法
import java.util.ArrayList; import java.util.List; public class StrTest { public static void main(St ...