【hdu2457】ac自动机 + dp
题目大意:
给你一个字符主串和很多病毒串,要求更改最少的字符使得没有一个病毒串是主串的子串。
题解:
ac自动机 + dp,用病毒串建好ac自动机,有毒的末尾flag置为true
构建fail指针时,如果fail指针所指节点flag=true,则将当前节点的flag也置为true。
用dp[i][j]表示长度为i的字符串匹配到j节点时最少的更改次数。
这样在ac自动机上跑,若该节点与原串的字符不同则+1
否则+0. 最后扫一遍取flag!=true的dp[len][...]的最小值。
code
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<algorithm>
#include<cstring>
using namespace std;
const int N = 55, L = 25, OO = 0x3f3f3f3f;
int n, dp[1005][N * L], tot, len, Case;
char t[L], s[1005];
struct node{
int trans[5], fail;
bool flag;
inline void clear(){
memset(trans, 0, sizeof trans);
flag = fail = 0;
}
}trie[N * L];
inline int getVal(char p){
if(p == 'A') return 1;
else if(p == 'C') return 2;
else if(p == 'G') return 3;
else return 4;
}
inline void insert(){
int pos = 1;
len = strlen(t + 1);
for(int i = 1; i <= len; i++){
if(!trie[pos].trans[getVal(t[i])])
trie[trie[pos].trans[getVal(t[i])] = ++tot].clear();
pos = trie[pos].trans[getVal(t[i])];
}
trie[pos].flag = true;
}
inline void buildFail(){
for(int i = 1; i <= 4; i++) trie[0].trans[i] = 1;
static int que[N * L], qn;
que[qn = 1] = 1;
for(int ql = 1; ql <= qn; ql++){
int u = que[ql], v, w;
for(int i = 1; i <= 4; i++){
v = trie[u].fail;
while(!trie[v].trans[i]) v = trie[v].fail;
v = trie[v].trans[i];
w = trie[u].trans[i];
if(w){
trie[w].fail = v, que[++qn] = w;
if(trie[v].flag)
trie[w].flag = true;
}
else trie[u].trans[i] = v;
}
}
}
inline void solve(){
len = strlen(s + 1);
for(int i = 1; i <= len; i++)
for(int j = 1; j <= tot; j++)
dp[i][j] = OO;
dp[0][1] = 0;
for(int i = 1; i <= len; i++)
for(int j = 1; j <= tot; j++){
if(dp[i - 1][j] == OO) continue;
for(int k = 1; k <= 4; k++){
int u = trie[j].trans[k];
if(trie[u].flag) continue;
dp[i][u] = min(dp[i][u], dp[i - 1][j] + (getVal(s[i]) != k));
}
}
int ans = OO;
for(int i = 1; i <= tot; i++){
if(dp[len][i] == OO) continue;
ans = min(ans, dp[len][i]);
}
printf("Case %d: %d\n", ++Case, ans < OO ? ans: -1);
}
int main(){
while(scanf("%d", &n), n){
trie[tot = 1].clear();
for(int i = 1; i <= n; i++){
scanf("%s", t + 1);
insert();
}
buildFail();
scanf("%s", s + 1);
solve();
}
return 0;
}
【hdu2457】ac自动机 + dp的更多相关文章
- HDU2457 DNA repair(AC自动机+DP)
题目一串DNA最少需要修改几个基因使其不包含一些致病DNA片段. 这道题应该是AC自动机+DP的入门题了,有POJ2778基础不难写出来. dp[i][j]表示原DNA前i位(在AC自动机上转移i步) ...
- POJ1625 Censored!(AC自动机+DP)
题目问长度m不包含一些不文明单词的字符串有多少个. 依然是水水的AC自动机+DP..做完后发现居然和POJ2778是一道题,回过头来看都水水的... dp[i][j]表示长度i(在自动机转移i步)且后 ...
- HDU2296 Ring(AC自动机+DP)
题目是给几个带有价值的单词.而一个字符串的价值是 各单词在它里面出现次数*单词价值 的和,问长度不超过n的最大价值的字符串是什么? 依然是入门的AC自动机+DP题..不一样的是这题要输出具体方案,加个 ...
- hdu 4117 GRE Words AC自动机DP
题目:给出n个串,问最多能够选出多少个串,使得前面串是后面串的子串(按照输入顺序) 分析: 其实这题是这题SPOJ 7758. Growing Strings AC自动机DP的进阶版本,主题思想差不多 ...
- hdu 2457(ac自动机+dp)
题意:容易理解... 分析:这是一道比较简单的ac自动机+dp的题了,直接上代码. 代码实现: #include<stdio.h> #include<string.h> #in ...
- HDU 2425 DNA repair (AC自动机+DP)
DNA repair Time Limit: 5000/2000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total ...
- HDU2296——Ring(AC自动机+DP)
题意:输入N代表字符串长度,输入M代表喜欢的词语的个数,接下来是M个词语,然后是M个词语每个的价值.求字符串的最大价值.每个单词的价值就是单价*出现次数.单词可以重叠.如果不止一个答案,选择字典序最小 ...
- tyvj P1519 博彩游戏(AC自动机+DP滚动数组)
P1519 博彩游戏 背景 Bob最近迷上了一个博彩游戏…… 描述 这个游戏的规则是这样的:每花一块钱可以得到一个随机数R,花上N块钱就可以得到一个随机序列:有M个序列,如果某个序列是产生的随机序列的 ...
- bzoj 1030 [JSOI2007]文本生成器(AC自动机+DP)
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=1030 [题意] 给n个小串,随机构造一个长为m的大串,一个串合法当且仅当包含一个或多个 ...
随机推荐
- Android 撕衣服(刮刮乐游戏)
项目简单介绍: 该项目为撕衣服,相似刮刮乐游戏 具体介绍: 用户启动项目后.载入一张图片,当用户点击图片的时候,点击的一片区域就会消失.从而显示出在这张图片以下的图片 这个小游戏相似与刮奖一样,刮开涂 ...
- log4j配置文件及nutch中的日志配置 分类: B1_JAVA 2015-02-17 10:58 483人阅读 评论(0) 收藏
吐槽几句,log4j的坑啊.... (1)CLASSPATH中不能有多个log4j的版本本,否则有有奇形怪状的NoSuchMethod, NoSuchFiled, NoClassDefineFound ...
- amazeui学习笔记--css(常用组件3)--按钮组Button-group
amazeui学习笔记--css(常用组件3)--按钮组Button-group 一.总结 1.按钮组用法:把一系列要使用的 .am-btn 按钮放入 .am-btn-group . 2.按钮工具栏: ...
- Javascript和jquery事件--鼠标事件的小结
1.鼠标事件的主要事件应该是mouseup, mousedown, mousewheel, mousemove, mouseover, moveout. <1>其中mouseup和mous ...
- 在 Windows 10 x64 上安装及使用 ab 工具的流程
本文转自:www.shuijingwanwq.com/2017/04/18/1568/ 1.基于AB测试工具进行高并发情形下的模拟测试,打开:http://httpd.apache.org/docs/ ...
- 希捷硬盘扩容软件-----DiscWizard
SeagateDiscWizard可为Seagate磁盘驱动器的使用提供便利.DiscWizard可帮助您迅速安装新的磁盘驱动器.并通过安装向导指导您在磁盘驱动器上完毕分区的创建和格式化. DiscW ...
- 学习C#修饰符:类修饰符和成员修饰符
C#修饰符之类修饰符:public.internal. partial.abstract.sealed.static C#修饰符之成员修饰符:public.protected.private.inte ...
- 百度ueditor vue项目应用 -- 图片上传源码修改
本文目的有两个,一.废掉单图上传,二.改造多图上传 大家都知道百度ueditor不是针对vue项目开发的,官方文档提供的源码包里有需要后端配置的接口,but到vue项目就不太好办了,网上有些文章也介绍 ...
- css实现悬浮效果的阴影
要实现的效果图: 图片.png 实现的代码: -webkit-box-shadow:0px 3px 3px #c8c8c8 ; -moz-box-shadow:0px 3px 3px #c8c8c8 ...
- 【u244】山地考察
Time Limit: 1 second Memory Limit: 128 MB [问题描述] 地质学家们打算考察一片山区.这片山区可分成m*n的网格,每个网格都有唯一的海拔高度,山区外围的海拔高度 ...