题意:给出n个资源,m个病毒,将资源串拼接成一个串,必须包含所有的资源串,可以重叠,但是不能包含病毒,问最小的长度为多少

题解:所有串建AC自动机。对以资源串结尾的结点跑bfs,求出到其他资源串结尾的最小距离。当前结点的fail结点不能入队列,因为当前结点读下一个字符可能会遇到禁止字符串,而fail结点读相同字符可能没有遇到禁止字符串...所以每次直接将nex的可行结点入队列。(fail结点不入队列,直接设为与当前结点距离为0应该也是可行的,然而WA了)很多时候我都开始质疑数据问题了...

如果当前字符串本身包含另一个资源串,那距离不应当是0吗?有一种解释是dp的时候经过 | 操作就可以了。

 #include <bits/stdc++.h>
#include <cstdio>
#include <iostream>
#include <algorithm>
#define ll long long
#define lson l, m, rt<<1
#define rson m+1, r, rt<<1|1
#define st first
#define nd second
#define mp make_pair
#define pii pair<int, int>
#define gg puts("gg");
#define local
//#define out1
using namespace std;
const int N = 6e4+;
const int inf = 0x3f3f3f3f;
int id(char c){
return c == '';
}
struct Tire{
int nex[N][], fail[N],end[N];
int root, L;
int node[], tot;
int newnode(){
nex[L][] = nex[L][] = -;
end[L] = ;
return L++;
}
void init(){
L = tot = ;
root = newnode();
}
void insert(char* s, int tag){
int now = root;
for(int i = ; s[i]; i++){
int p = id(s[i]);
if(nex[now][p] == -)
nex[now][p] = newnode();
now = nex[now][p];
}
if(tag >= )
end[now] |= <<tag;
else end[now] = -;
}
void build(){
queue<int> Q;
fail[root] = root;
for(int i = ; i < ; i++){
int& u = nex[root][i];
if(u == -)
u = root;
else{
fail[u] = root;
Q.push(u);
}
}
while(!Q.empty()){
int now = Q.front();
Q.pop();
for(int i = ; i < ; i++){
int& u = nex[now][i];
if(u == -)
u = nex[ fail[now] ][i];
else{
fail[u] = nex[ fail[now] ][i];
end[u] |= end[ fail[u] ];
Q.push(u);
}
}
}
}
};
Tire ac;
char s[N];
void gmin(int& a, int b){
if(a > b) a = b;
}
int dis[N], ve[N], tot;
int w[][]; void bfs(int node, int i){
memset(dis, 0x3f, sizeof(dis));
queue<int> Q;
Q.push(node); dis[node] = ;
int s = ; #ifdef out1
int j = node;
while(ac.fail[j] != ac.root){
j = ac.fail[j];
dis[j] = ;
}
dis[j] = ;
#endif while(!Q.empty()){
int sz = Q.size();
s++;
while(sz--){
int f = Q.front(); Q.pop();
for(int j = ; j < ; j++){
int to = ac.nex[f][j];
if(dis[to] < inf||ac.end[to] < ) continue ;
Q.push(to);
dis[to] = s; #ifdef out1
while(ac.fail[to] != ac.root&&dis[ac.fail[to]] == inf){
to = ac.fail[to];
dis[to] = s;
}
#endif
}
}
}
for(int j = ; j < tot; j++)
w[i][j] = dis[ ve[j] ];
}
int dp[][<<];
int main(){
#ifdef locl
freopen("in", "r", stdin);
#ifdef out1
freopen("out1", "w", stdout);
#else
freopen("out2", "w", stdout);
#endif // out1
#endif // locl
int n, m, t, ca = ;
while(~scanf("%d%d", &n, &m), n+m){
ac.init(); memset(w, 0x3f, sizeof(w));
for(int i = ; i < n; i++){
w[i][i] = ;
scanf("%s", s);
ac.insert(s, i);
}
for(int i = ; i < m; i++){
scanf("%s", s);
ac.insert(s, -);
}
ac.build(); tot = ;
for(int i = ; i < ac.L; i++)
if(!i||ac.end[i] > ) ve[tot++] = i;
memset(w, 0x3f, sizeof(w));
for(int i = ; i < tot; i++)
bfs(ve[i], i);
// for(int i = 0; i < tot; i++){
// for(int j = 0; j < tot; j++)
// printf("%10d ", w[i][j]);
// puts("");
// } memset(dp, 0x3f, sizeof(dp));
for(int i = ; i < tot; i++)
dp[i][ ac.end[ve[i]] ] = w[][i];
for(int i = ; i < (<<n); i++){
for(int j = ; j < tot; j++){
if( dp[j][i] < 0x3f3f3f3f ) for(int k = ; k < tot; k++)
if(j != k&&w[j][k] < 0x3f3f3f3f)
gmin(dp[k][i|ac.end[ve[k]]], dp[j][i]+w[j][k]);
}
} // for(int i = 0; i < n; i++)
// for(int j = 0; j < (1<<n); j++)
// printf("%d %d: %d\n", i, j, dp[i][j]); int ans = inf;
for(int i = ; i < tot; i++)
ans = min(ans, dp[i][ (<<n)- ]);
printf("%d\n", ans);
}
return ;
}
/*
2 2
1110
0111
101
1001 3 3
0001000
00100
010
11
111
1111
*/

HDU3247 AC自动机+dp的更多相关文章

  1. POJ1625 Censored!(AC自动机+DP)

    题目问长度m不包含一些不文明单词的字符串有多少个. 依然是水水的AC自动机+DP..做完后发现居然和POJ2778是一道题,回过头来看都水水的... dp[i][j]表示长度i(在自动机转移i步)且后 ...

  2. HDU2296 Ring(AC自动机+DP)

    题目是给几个带有价值的单词.而一个字符串的价值是 各单词在它里面出现次数*单词价值 的和,问长度不超过n的最大价值的字符串是什么? 依然是入门的AC自动机+DP题..不一样的是这题要输出具体方案,加个 ...

  3. HDU2457 DNA repair(AC自动机+DP)

    题目一串DNA最少需要修改几个基因使其不包含一些致病DNA片段. 这道题应该是AC自动机+DP的入门题了,有POJ2778基础不难写出来. dp[i][j]表示原DNA前i位(在AC自动机上转移i步) ...

  4. hdu 4117 GRE Words AC自动机DP

    题目:给出n个串,问最多能够选出多少个串,使得前面串是后面串的子串(按照输入顺序) 分析: 其实这题是这题SPOJ 7758. Growing Strings AC自动机DP的进阶版本,主题思想差不多 ...

  5. hdu 2457(ac自动机+dp)

    题意:容易理解... 分析:这是一道比较简单的ac自动机+dp的题了,直接上代码. 代码实现: #include<stdio.h> #include<string.h> #in ...

  6. HDU 2425 DNA repair (AC自动机+DP)

    DNA repair Time Limit: 5000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total ...

  7. HDU2296——Ring(AC自动机+DP)

    题意:输入N代表字符串长度,输入M代表喜欢的词语的个数,接下来是M个词语,然后是M个词语每个的价值.求字符串的最大价值.每个单词的价值就是单价*出现次数.单词可以重叠.如果不止一个答案,选择字典序最小 ...

  8. tyvj P1519 博彩游戏(AC自动机+DP滚动数组)

    P1519 博彩游戏 背景 Bob最近迷上了一个博彩游戏…… 描述 这个游戏的规则是这样的:每花一块钱可以得到一个随机数R,花上N块钱就可以得到一个随机序列:有M个序列,如果某个序列是产生的随机序列的 ...

  9. bzoj 1030 [JSOI2007]文本生成器(AC自动机+DP)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=1030 [题意] 给n个小串,随机构造一个长为m的大串,一个串合法当且仅当包含一个或多个 ...

随机推荐

  1. JAVA字段的初始化规律

    JAVA字段的初始化规律 1.类的构造方法 (1)“构造方法”,也称为“构造函数”,当创建一个对象时,它的构造方法会被自动调用.构造方法与类名相同,没有返回值. (2)如果类没有定义构造函数,Java ...

  2. 64位系统安装ODBC驱动的方法

    为了更充分的利用硬件资源,我想很多人都开使用64位操作系统了,同时你可以也发现了在64位操作系统上ODBC的驱动找不到了,所以ODBC的东西都没法用了. 因为2007以前版本的Office只有32位版 ...

  3. linux命令总结2

    昨天继续对239进行挖掘,想把运营登录浮层的示例页面放在这台测试机上,供大家使用,结果在配置apache时出现了问题,无论怎样,页面都是403 Forbidden,最后终于被露颖同学经过2个小时的努力 ...

  4. 如果在遨游浏览器里设置Bing(必应)搜索为默认搜索

    今天刚装了遨游浏览器,发现搜索引擎列表里没有Bing(必应)搜索的选项,就自己DIY了下. 步骤: 1. 在遨游搜索引擎列表管理里,添加一个新的搜索引擎项: Name:填写上”Bing(必应)“ (这 ...

  5. CSS选择器及其优先级

    一:一些普通的选择器 <!DOCTYPE html> <html> <head lang="en"> <meta charset=&quo ...

  6. github for windows回滚到某一个版本,

    建议先学会使用git命令再学GUI版的git,git本来就是命令行程序,GUI本质就是执行一些命令.仅从一些icon和单词去理解一些操作难免会有偏差.而反过来,熟悉命令会更好地理解GUI操作.想要回滚 ...

  7. boost库学习之开篇

    本系列文章使用boost_1.58.0版本. 一.欢迎使用boost C++库 boost致力于提供一个免费的.便携的源代码级的库. 我们重视那些与C++标准一起工作良好的库.boost库将要成为一个 ...

  8. Python类的定义与使用

    #! /usr/bin/python # Filename:objvar.py class Person: '''Represents a person.''' population = 0 def ...

  9. js调用MVC3自带js验证

    验证: if ($(this).is("form")) {                        return $(this).validate().checkForm() ...

  10. mysql 导入导出的几个常用参数

    导出命令: mysqldump -t --skip-extended-insert -utest -p testdb tableA > testdb_tableA.sql 参数说明: -t: 仅 ...