AC自动机+状压DP

  首先对所有串建AC自动机,然后对于每个资源串,算出从串末走到其他资源串末所需的距离(中途避开非法点)

  也就是算出两两间的距离。。。然后就变成旅行商问题了。

  计算距离的时候要考虑一下。。直接从末尾走的话可能求出来的并不是最短路

  所以要从fail和ch(就是虚边和实边)一起转移,走实边的话加进队列,虚边只能更新距离(不然可能非法)

  但似乎正确性存疑?。。可能存在 三个资源串,分别按照最短距离拼在一起后会包含病毒串 的情况。。。

  反正我的程序会炸,但网上一些标程没事= =。。。另外本题数据极弱,随机数据就能艹翻一大坨标程系列

 #include<cstdio>
#include<iostream>
#include<cstring>
#define next NEXT
using namespace std;
const int maxn=,inf=;
int ch[maxn][],next[maxn][],fail[maxn],tot;
bool gg[maxn],uu[maxn];
int f1[maxn];
char s[maxn],mp[][];int len[],ed[];
int d[][],dis[maxn],d1[maxn];
int f[][],two[];
int dl[maxn],l,r;
int i,j,k,n,m; inline void getf1(int x){
int i,p=;f1[]=;
for(i=;i<=len[x];i++){
while(p&&mp[x][i]!=mp[x][p+])p=f1[p];
p+=mp[x][i]==mp[x][p+],f1[i]=p;
}
}
inline bool check(int x,int y){
int i,p=;
for(i=;i<=len[y];i++){
while(p&&mp[y][i]!=mp[x][p+])p=f1[p];
p+=mp[y][i]==mp[x][p+];
if(p==len[x])return ;
}
return ;
} inline void trie(int n,int id){
int i,p=;
for(i=;i<=n;i++){
s[i]-='';
if(!ch[p][s[i]])ch[p][s[i]]=++tot,p=tot;
else p=ch[p][s[i]];
}
if(!id)gg[p]=;else ed[id]=p;//printf(" gg:%d\n",p);
}
inline void getfail(){
int i,now,p,j;l=,r=,dl[]=;
while(l<r){
now=dl[++l];
for(i=;i<;/*printf("%d %d next:%d\n",now,i,next[now][i]),*/i++)if(ch[now][i]){
dl[++r]=j=next[now][i]=ch[now][i];
for(p=fail[now];p&&!ch[p][i];p=fail[p]);
if(!now)fail[j]=;else fail[j]=ch[p][i];
gg[j]|=gg[fail[j]];
}else{
for(p=fail[now];p&&!ch[p][i];p=fail[p]);
next[now][i]=ch[p][i];
}
}
for(i=;i<=tot;i++)if(gg[i])
for(j=/*,printf("!! gg:%d\n",i)*/;j<;j++)if(ch[i][j])gg[ch[i][j]]=;
}
inline void bfs(int s){
memset(dis,,(tot+)<<);memset(uu,,tot+);
int l=,r=,i,now,j;
if(!gg[s])dl[++r]=s,dis[s]=d1[s]=,uu[s]=;
for(i=fail[s];i;i=fail[i])if(!gg[i])dis[i]=;
while(l<r){
now=dl[++l];
for(i=;i<;i++)if(!gg[next[now][i]]&&!uu[next[now][i]]){
dl[++r]=next[now][i];uu[dl[r]]=;d1[dl[r]]=d1[now]+;
if(dis[dl[r]]==-)dis[dl[r]]=d1[dl[r]];
for(j=fail[dl[r]];j;j=fail[j])if(dis[j]==-)dis[j]=dis[dl[r]];
// if(s==16)printf("walk:%d->%d\n",now,next[now][i]);
}
}
// if(s==16)
// for(i=0;i<=tot;i++)printf(" %d->%d %d\n",s,i,dis[i]);
}
int main(){
freopen("in.txt","r",stdin);
for(i=two[]=;i<=;i++)two[i]=two[i-]<<;
for(scanf("%d%d",&n,&m);n&&m;scanf("%d%d",&n,&m)){
memset(ch,,(tot+)<<),memset(next,,(tot+)<<),memset(fail,,(tot+)<<),
memset(gg,,tot+),tot=; for(i=;i<=n;i++)
scanf("%s",mp[i]+),mp[i][]='%',len[i]=strlen(mp[i])-;
for(i=;i<n;i++)for(j=i+;j<=n;j++)if(len[i]>len[j])
swap(len[i],len[j]),swap(mp[i],mp[j]); for(i=;i<n;i++){
getf1(i);
for(j=i+;j<=n;j++)if(check(i,j)){gg[i]=;/*printf("%d in %d\n",i,j);*/break;}
}
int n1=;
for(i=;i<=n;i++)if(!gg[i])n1++,memcpy(mp[n1],mp[i],len[i]+),len[n1]=len[i];//,printf("%s\n",mp[n1]);
memset(gg,,n+); for(i=;i<=n1;i++)memcpy(s,mp[i],len[i]+),trie(len[i],i);
for(i=;i<=m;i++)scanf("%s",s+),s[]='%',trie(strlen(s)-,);
getfail(); for(i=;i<=n1;puts(""),i++){
bfs(ed[i]);
for(j=;j<=n1;/*printf("%s-->%s %d\n",mp[i]+1,mp[j]+1,d[i][j]),*/j++)if(dis[ed[j]]!=-)d[i][j]=dis[ed[j]];else d[i][j]=inf;
} int mx=<<n1;
for(i=;i<mx;i++)memset(f[i],,(n1+)<<);
for(i=;i<=n1;i++)f[two[i-]][i]=len[i];
for(i=;i<mx;i++){
for(j=;j<=n1;j++)if(i&two[j-])
for(k=;k<=n1;k++)if(k!=j&&(i&two[k-])&&d[k][j]<inf)
f[i][j]=min(f[i][j],f[i^two[j-]][k]+d[k][j]);
}
int ans=inf;
for(i=;i<=n1;i++)ans=min(ans,f[mx-][i]);
}
return ;
}

[HDU3247]Resource Archiver的更多相关文章

  1. HDU3247 Resource Archiver —— AC自动机 + BFS最短路 + 状压DP

    题目链接:https://vjudge.net/problem/HDU-3247 Resource Archiver Time Limit: 20000/10000 MS (Java/Others)  ...

  2. HDU-3247 Resource Archiver(AC自动机+BFS)

    Description Great! Your new software is almost finished! The only thing left to do is archiving all ...

  3. HDU3247 Resource Archiver (AC自动机+spfa+状压DP)

    Great! Your new software is almost finished! The only thing left to do is archiving all your n resou ...

  4. HDU3247 Resource Archiver(AC自动机+BFS+DP)

    题目,求最短的包含所有n个DNA片段且不包含任何一个病毒片段的序列. 容易用所有DNA片段和病毒片段建一个AC自动机,构造fail时处理一下各个结点后缀是DNA或者病毒的情况,然后dp[S][u]表示 ...

  5. 【HDU3247】 Resource Archiver(DP+AC自动机+最短路)

    Resource Archiver Time Limit: 10000MS   Memory Limit: 100000KB   64bit IO Format: %I64d & %I64u ...

  6. HDU 3247 Resource Archiver (AC自己主动机 + BFS + 状态压缩DP)

    题目链接:Resource Archiver 解析:n个正常的串.m个病毒串,问包括全部正常串(可重叠)且不包括不论什么病毒串的字符串的最小长度为多少. AC自己主动机 + bfs + 状态压缩DP ...

  7. HDU - 3247 Resource Archiver (AC自动机,状压dp)

    \(\quad\)Great! Your new software is almost finished! The only thing left to do is archiving all you ...

  8. HDU 3247 Resource Archiver (AC自动机+BFS+状压DP)

    题意:给定 n 个文本串,m个病毒串,文本串重叠部分可以合并,但合并后不能含有病毒串,问所有文本串合并后最短多长. 析:先把所有的文本串和病毒都插入到AC自动机上,不过标记不一样,可以给病毒标记-1, ...

  9. Resource Archiver HDU - 3247 AC自动机+BFS+状压

    题意: 给出n个资源串,m个病毒串,现在要如何连接资源串使得不含病毒串(可以重叠,样例就是重叠的). 题解: 这题的套路和之前的很不同了,之前的AC自动机+DP的题目一般都是通过teir图去转移, 这 ...

随机推荐

  1. iOS 友盟推送,应用内推送启动图推送闪动黑屏,插屏推送方法报错

    以前都是用的极光推送,应公司需求要求使用友盟推送,为了以后是有分享都适用,,, 友盟推送文档,下载demo 感觉比极光用着要简单顺手 一切就绪后,开始发送消息测试,,,,,搞了半天没有发过来消息 原来 ...

  2. ios 访问隐私信息 info.plist 中的字段

    1.iOS10相册相机闪退bug: iOS10系统下调用系统相册,相机功能,遇到闪退的情况,描述如下: This app has crashed because it attempted to acc ...

  3. lesson - 10 课程笔记

    CTRL+C CTRL+D :前者用于结束一个程序,后者用于结束终端输入. --符号 *:匹配任意长度的任意字符 ?:匹配任意一个字符 #: shell 中表注释 \: 脱意符号 []:任意属于字符组 ...

  4. Cleaner, more elegant, and harder to recognize (msdn blog)

    It appears that some people interpreted the title of one of my rants from many months ago, "Cle ...

  5. appium 解锁九宫格

    很多人在自动化的过程中,对解锁9宫格有很多麻烦,特别是app上的有些整个放在整个view中,这就给我们测试解锁九宫格带来问题了,笔者尝试了去解决,但是都没有找到一个很好的方案,那么我就试着先去通过安卓 ...

  6. Python 项目实践三(Web应用程序)第三篇

    接着上节的继续学习,现在要显示所有主题的页面 有了高效的网页创建方法,就能专注于另外两个网页了:显示全部主题的网页以及显示特定主题中条目的网页.所有主题页面显示用户创建的所有主题,它是第一个需要使用数 ...

  7. JAVA个人理解

    为了找到别人写的好文章,先分享下自己的知识,找找感觉路线. 学java前接触的c,后来转向java.第一个照面理解的就是面向对象,没想到让我想了好多年.当时有个负责任的老师说面向对象这个词具体释义众说 ...

  8. thinkinginjava学习笔记03_基本语法

    由于java是c系语言,基本保留c语言的所有基本操作,就快速过一下: java中的基本操作符仅仅对基本类型有效:=.==.!=对所有对象有效(值传递),String类支持+.+=; 在对基本数据进行算 ...

  9. windows的ReactNative挖坑一分钟爬坑一小时

    其实开发并不需要Android Studio来开发,因为命令行都是要自己手打的,所以就开始了我的挖坑爬坑之旅 首先安装React Native要用到的git.nodejs等等这里不讲了,主要讲在手机上 ...

  10. Python装饰器的解包装(unwrap)

    在Python 3.4 中,新增一个方法unwrap,用于将被装饰的函数,逐层进行解包装. inspect.unwrap(func, *, stop=None) unwrap方法接受两个参数:func ...