hdu3247Resource Archiver (AC自动机+最短路+状压dp)
Time Limit: 20000/10000 MS (Java/Others) Memory Limit: 100000/100000 K (Java/Others)
Total Submission(s): 2302 Accepted Submission(s): 724
Wait a minute… you realized that it isn’t as easy as you thought. Think about the virus killers. They’ll find your software suspicious, if your software contains one of the m predefined virus codes. You absolutely don’t want this to happen.
Technically, resource files and virus codes are merely 01 strings. You’ve already convinced yourself that none of the resource strings contain a virus code, but if you make the archive arbitrarily, virus codes can still be found somewhere.
Here comes your task (formally): design a 01 string that contains all your resources (their occurrences can overlap), but none of the virus codes. To make your software smaller in size, the string should be as short as possible.
and virus codes are all non-empty 01 strings without spaces inside. Each resource is at most 1000 characters long. The total length of all virus codes is at most 50000. The input ends with n = m = 0.
1110
0111
101
1001
0 0
题意:给你n个01串和m个01串,让你构造一个最短的字符串要包含前n个,但不能包含m中的任意一个。
思路:先把(n+m)个串都放进trie图中,那么n(n<=10)个节点是否已经经过的状态可以用二进制表示出来,然后把m个串的尾节点都设为危险节点,然后进行dp。容易想到的dp是用dp[j][state]表示走到第j个节点,当前n个串的储存情况为state所需要的最少步数,但是这样的时间复杂度是60000*1024*10超时了。所以我们会发现其实在dp过程中,很多节点都是没有用的,即对n个点状态没有影响,只有到n个字符串尾节点对应的trie图节点,state状态才会改变,所以我们考虑先在根节点,n个尾节点在trie图中对应的节点两两之间求最短路即最少步数,然后就在这n个点中状压dp就行了。
#include<iostream>
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<math.h>
#include<vector>
#include<map>
#include<set>
#include<queue>
#include<stack>
#include<string>
#include<algorithm>
using namespace std;
typedef long long ll;
#define inf 99999999
#define pi acos(-1.0)
#define maxnode 60050
int dp[20][1030],dist[20][20],vis[maxnode];
int n,m;
map<int,int>jiedianhao;
map<int,int>chuanhao;
map<int,int>::iterator it;
struct trie{
int sz,root,val[maxnode],next[maxnode][2],fail[maxnode];
int q[1111111],q1[1111111][2];
void init(){
int i;
sz=root=0;
val[0]=0;
for(i=0;i<2;i++){
next[root][i]=-1;
}
jiedianhao[0]=0;
chuanhao[0]=0;
}
int idx(char c){
return c-'0';
}
void charu(char *s,int num){
int i,j,u=0;
int len=strlen(s);
for(i=0;i<len;i++){
int c=idx(s[i]);
if(next[u][c]==-1){
sz++;
val[sz]=0;
next[u][c]=sz;
u=next[u][c];
for(j=0;j<2;j++){
next[u][j]=-1;
}
}
else{
u=next[u][c];
}
}
if(num==-1)val[u]=-1;
else{
jiedianhao[num]=u;
chuanhao[u]=num;
val[u]|=(1<<num-1);
}
}
void build(){
int i,j;
int front,rear;
front=1;rear=0;
for(i=0;i<2;i++){
if(next[root][i]==-1 ){
next[root][i]=root;
}
else{
fail[next[root][i] ]=root;
rear++;
q[rear]=next[root][i];
}
}
while(front<=rear){
int x=q[front];
if(val[fail[x] ]==-1 )val[x]=-1;
else val[x]|=val[fail[x] ];
front++;
for(i=0;i<2;i++){
if(next[x][i]==-1){
next[x][i]=next[fail[x] ][i];
}
else{
fail[next[x][i] ]=next[fail[x] ][i];
rear++;
q[rear]=next[x][i];
}
}
}
}
void debug(){
int i,j;
for(j=0;j<=sz;j++){
printf("--->%d %d %d %d\n",j,val[j],next[j][0],next[j][1]);
}
}
void bfs()
{
int i,j,state,x,t,tt,xx;
memset(dist,0,sizeof(dist));
for(i=0;i<=n;i++){
for(j=0;j<=sz;j++)vis[j]=0;
vis[jiedianhao[i] ]=1;
if(i!=0)state=(1<<(i-1) );
else state=0;
int front=1;
int rear=0;
rear++;
q1[rear][0]=jiedianhao[i];q1[rear][1]=0;
while(front<=rear){
x=q1[front][0];t=q1[front][1];
front++;
for(j=0;j<2;j++){
xx=next[x][j];
tt=t+1;
if(vis[xx])continue;
if(val[xx]==-1)continue;
vis[xx]=1;
if(val[xx]!=0){
state|=val[xx];
dist[i][chuanhao[xx] ]=t+1;
if(state==(1<<n)-1)break;
}
rear++;
q1[rear][0]=xx;q1[rear][1]=tt;
}
if(state==(1<<n)-1)break;
}
}
}
void solve(){
int i,j,state,state1;
for(i=0;i<=n;i++){
for(state=0;state<(1<<n);state++ ){
dp[i][state]=inf;
}
}
dp[0][0]=0;
for(state=0;state<(1<<n);state++){ //这个状压dp和tsp问题中的有些不同,这里是由已知的状态去推未知的状态,因为可能一个字符串包含其他字符串。
for(i=0;i<=n;i++){
if(dp[i][state]!=inf){
for(j=1;j<=n;j++){
dp[j][state|val[jiedianhao[j] ] ]=min(dp[j][state|val[jiedianhao[j] ] ],dp[i][state ]+dist[i][j] );
}
}
}
}
int minx=inf;
for(i=0;i<=n;i++){
minx=min(minx,dp[i][(1<<n)-1]);
}
printf("%d\n",minx);
}
}ac;
char s[1005],str[50050];
int main()
{
int i,j;
while(scanf("%d%d",&n,&m)!=EOF)
{
if(n==0 && m==0)break;
jiedianhao.clear();
chuanhao.clear();
ac.init();
for(i=1;i<=n;i++){
scanf("%s",s);
ac.charu(s,i);
}
for(i=1;i<=m;i++){
scanf("%s",str);
ac.charu(str,-1);
}
ac.build();
ac.bfs();
ac.solve();
}
return 0;
}
hdu3247Resource Archiver (AC自动机+最短路+状压dp)的更多相关文章
- HDU2825 Wireless Password 【AC自动机】【状压DP】
HDU2825 Wireless Password Problem Description Liyuan lives in a old apartment. One day, he suddenly ...
- 【AC自动机】【状压dp】【滚动数组】hdu6086 Rikka with String
给你m个01串,问你有多少个长度为2L的01串,满足前半段倒置取反后等于后半段,并且包含所有的m个01串. 考虑单词完全在中线前面或者后面的情况,直接将单词及其倒置取反插入AC自动机,AC自动机每个结 ...
- 【AC自动机】【状压dp】hdu2825 Wireless Password
f(i,j,S)表示当前字符串总长度为i,dp到AC自动机第j个结点,单词集合为S时的方案数. 要注意有点卡常数,注意代码里的注释. #include<cstdio> #include&l ...
- HDU3247 Resource Archiver —— AC自动机 + BFS最短路 + 状压DP
题目链接:https://vjudge.net/problem/HDU-3247 Resource Archiver Time Limit: 20000/10000 MS (Java/Others) ...
- HDU 4568 Hunter 最短路+状压DP
题意:给一个n*m的格子,格子中有一些数,如果是正整数则为到此格子的花费,如果为-1表示此格子不可到,现在给k个宝藏的地点(k<=13),求一个人从边界外一点进入整个棋盘,然后拿走所有能拿走的宝 ...
- 最短路+状压DP【洛谷P3489】 [POI2009]WIE-Hexer
P3489 [POI2009]WIE-Hexer 大陆上有n个村庄,m条双向道路,p种怪物,k个铁匠,每个铁匠会居住在一个村庄里,你到了那个村庄后可以让他给你打造剑,每个铁匠打造的剑都可以对付一些特定 ...
- 【BZOJ1097】[POI2007]旅游景点atr 最短路+状压DP
[BZOJ1097][POI2007]旅游景点atr Description FGD想从成都去上海旅游.在旅途中他希望经过一些城市并在那里欣赏风景,品尝风味小吃或者做其他的有趣的事情.经过这些城市的顺 ...
- ACM-ICPC2018南京网络赛 AC Challenge(一维状压dp)
AC Challenge 30.04% 1000ms 128536K Dlsj is competing in a contest with n (0 < n \le 20)n(0<n ...
- 【CodeVS2800】 送外卖 最短路+状压DP
首先求出各点之间的最短路,floyed即可,注意是0-n. 然后考虑状压,f[i][j]表示状态为i时访问j点时的最短路和,1表示访问,0表示未访问,然后第j个点所在的位置就是(1<<j) ...
随机推荐
- 【Java基础】网络编程
网络编程 网络编程概述 网络编程的目的:直接或简洁地通过网络协议与其他计算机实现数据交换,进行通讯. 网络编程的两个主要问题: 如果准确地定位网络上一台或多台主机,并定位主机上的特定应用: 找到主机后 ...
- 【C++】《Effective C++》第一章
第一章 让自己习惯C++ C++是一个威力强大的语言,带着众多特性,但是在你可以驾驭其威力并有效运用其特性之前,你必须先习惯C++的办事方式. 条款01:视C++为一个语言联邦 如今的C++已经是个多 ...
- golang遍历时修改被遍历对象
目录 前言 遍历切片 遍历map 总结 前言 很多时候需要将遍历对象中去掉某些元素,或者往遍历对象中添加元素,这时候就需要小心操作了. 对于go语言中的一些注意事项我做了总结和示例,留下点笔记. 遍历 ...
- 【Git】3、创建Git版本库、配置Git仓库用户邮箱信息
初识Git 文章目录 初识Git 1.创建Git版本库 认识.git 2.基础配置 2.1.查看配置信息 2.2.配置昵称邮箱信息 2.3.修改配置信息 1.通过命令行 2.通过修改配置文件. 修改全 ...
- 【Oracle】 并行查询
所谓并行执行,是指能够将一个大型串行任务(任何DML,一般的DDL)物理的划分为叫多个小的部分,这些较小的部分可以同时得到处理.何时使用并行执行:1.必须有一个非常大的任务 2.必须有充足的资源(CP ...
- 【Oracle】密码文件相关
Oracle数据库的orapwd命令,主要用来建立密码(口令)文件. 一.查看帮助信息 [oracle@oracle11g dbs]$ orapwd Usage: orapwd file=<fn ...
- 【Oracle】查看某个角色中有什么权限
select * from role_sys_privs where role='DBA'; 查看dba都有什么系统权限 select * from role_sys_privs where rol ...
- 超精讲-逐例分析CS:LAB2-Bomb!(上)
0. 环境要求 关于环境已经在lab1里配置过了这里要记得安装gdb 安装命令 sudo yum install gdb 实验的下载地址 http://csapp.cs.cmu.edu/3e/labs ...
- cts project的创建修改和删除
事务码:SPRO_ADMIN进入 项目管理界面,点击工具栏创建项目(F5),弹出对话框,输入项目名称,回车确定. 标题中输入项目的描述.点击保存.如图: 点击图片放大 注:要想此项目在CTS建立请求的 ...
- javascript判断浏览器访问,刷新,返回
话不多说,直接上 if (window.performance.navigation.type === 0/* 正常访问 */) { // 你要干的事 } else if (window.perfor ...