我有病吧……明明直接枚举是否匹配就可以非要写hash,然后果然冲突了(……我个非酋居然还敢用hash

设f[s][i]为已选串状态为s并且最后一个串是i,还有预处理出g[i][j]表示最长有长为g[i][j]的i串后缀等于j串前缀这里,直接暴力匹配即可……

然后注意到比较麻烦的事要求字典序最小,但是因为空间限制我们又不能给每个f存一个串,所以我们设t[s][i][j]为状态使f[s][i]长度最小且字典序最小的选串顺序的第j个选的是那个串,这个更新f[s][i]的时候直接更新,注意如果有等于当前f[s][i]的情况,要分别把旧的t[s][i]和当前情况对应的两个串都求出来比一下字典序

长度dp部分随便转移一下就好

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
const int N=15;
int n,f[5005][N],t[5005][N][N],g[N][N],len[N],ans=1e9,b[N];
bool v[N];
char c[N][55],s1[1005],s2[1005],a[1005];
struct qwe
{
int l;
char c[55];
};
bool cmp(const qwe &a,const qwe &b)
{
for(int i=1;i<=min(a.l,b.l);i++)
if(a.c[i]!=b.c[i])
return a.c[i]<b.c[i];
return a.l<b.l;
}
bool ok(int i,int j,int l)
{
for(int k=1;k<=l;k++)
if(c[i][len[i]-l+k]!=c[j][k])
return 0;
return 1;
}
int main()
{
scanf("%d",&n);
for(int i=1;i<=n;i++)
{
scanf("%s",c[i]+1);
len[i]=strlen(c[i]+1);
b[i]=1<<(i-1);
}
for(int i=1;i<=n;i++)
for(int j=1;j<=n;j++)
for(int l=min(len[i],len[j]);l>=0;l--)
if(ok(i,j,l))
{
g[i][j]=l;
break;
}
// dfs(1,0);
memset(f,0x3f,sizeof(f));
int inf=f[0][0];
for(int i=1;i<=n;i++)
f[b[i]][i]=len[i],t[b[i]][i][1]=i;
for(int s=1,le=(1<<n)-1;s<=le;s++)
{
int con=0;
for(int i=1;i<=n;i++)
if(s&b[i])
con++;
for(int i=1;i<=n;i++)
if(s&b[i])
{
for(int j=1;j<=n;j++)
if(!(s&b[j]))
{
int nw=f[s][i]+len[j]-g[i][j],w=(s|b[j]);
if(nw<f[w][j])
{
f[w][j]=nw;
for(int k=1;k<=con;k++)
t[w][j][k]=t[s][i][k];
t[w][j][con+1]=j;
}
else if(nw==f[w][j])
{
int t1=0,t2=0,fl=0;
for(int k=1;k<=con+1;k++)
for(int l=1+g[t[w][j][k-1]][t[w][j][k]];l<=len[t[w][j][k]];l++)
s1[++t1]=c[t[w][j][k]][l];
for(int k=1;k<=con;k++)
for(int l=1+g[t[s][i][k-1]][t[s][i][k]];l<=len[t[s][i][k]];l++)
s2[++t2]=c[t[s][i][k]][l];
for(int l=1+g[t[s][i][con]][j];l<=len[j];l++)
s2[++t2]=c[j][l];
for(int k=1;k<=nw;k++)
{
if(s1[k]<s2[k])
break;
if(s1[k]>s2[k])
{
fl=1;
break;
}
}
if(fl)
{
for(int k=1;k<=con;k++)
t[w][j][k]=t[s][i][k];
t[w][j][con+1]=j;
}
}
}
}
}
// printf("%d\n",ans);
int t1=0,w=(1<<n)-1,ans=1e9;
for(int j=1;j<=n;j++)
{
if(f[w][j]<ans)
{
ans=f[w][j];
int t1=0;
for(int k=1;k<=n;k++)
for(int l=1+g[t[w][j][k-1]][t[w][j][k]];l<=len[t[w][j][k]];l++)
a[++t1]=c[t[w][j][k]][l];
}
else if(f[w][j]==ans)
{
int t1=0,fl=0;
for(int k=1;k<=n;k++)
for(int l=1+g[t[w][j][k-1]][t[w][j][k]];l<=len[t[w][j][k]];l++)
s1[++t1]=c[t[w][j][k]][l];
for(int k=1;k<=ans;k++)
{
if(a[k]<s1[k])
break;
if(a[k]>s1[k])
{
fl=1;
break;
}
}
if(fl)
for(int k=1;k<=ans;k++)
a[k]=s1[k];
}
}
for(int i=1;i<=ans;i++)
printf("%c",a[i]);
return 0;
}

bzoj 1195: [HNOI2006]最短母串【状压dp】的更多相关文章

  1. bzoj 1195: [HNOI2006]最短母串 爆搜

    1195: [HNOI2006]最短母串 Time Limit: 10 Sec  Memory Limit: 32 MBSubmit: 894  Solved: 288[Submit][Status] ...

  2. BZOJ 1195: [HNOI2006]最短母串

    1195: [HNOI2006]最短母串 Time Limit: 10 Sec  Memory Limit: 32 MBSubmit: 1346  Solved: 450[Submit][Status ...

  3. bzoj 1195 [HNOI2006]最短母串 bfs 状压 最短路 AC自动机

    LINK:最短母串 求母串的问题.不适合SAM. 可以先简化问题 考虑给出的n个字符串不存在包含关系. 那么 那么存在的情况 只可能有 两个字符串拼接起来能表示另外一个字符串 或者某个字符串的后缀可以 ...

  4. BZOJ 1195 [HNOI2006]最短母串 (Trie图+状压+bfs最短路)

    BZOJ1195 LOJ10061 题目大意:给你$n$个模式串,求一个最短且字典序最小的文本串并输出这个串,$n<=12,len<=50$ 首先对所有模式串构造$Trie$图,$Trie ...

  5. BZOJ 1195: [HNOI2006]最短母串 AC自动机+状压+搜索

    思路比较直接. 由于 $n$ 很小,直接定义 $f[i][j]$ 表示当前在自动机中的节点 $i,$ 被覆盖串的集合为 $j$ 的方案数. #include <bits/stdc++.h> ...

  6. 【刷题】BZOJ 1195 [HNOI2006]最短母串

    Description 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T的子串. Input 第一行是一个正整数n(n<=12) ...

  7. 【状态压缩dp】1195: [HNOI2006]最短母串

    一个清晰的思路就是状压dp:不过也有AC自动机+BFS的做法 Description 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T ...

  8. 1195: [HNOI2006]最短母串 - BZOJ

    Description 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T的子串.Input 第一行是一个正整数n(n<=12), ...

  9. 1195: [HNOI2006]最短母串

    思路:好像以前谁问过我这题...  状个压就好啦, 把包含在其他串中的字符串删掉, 预处理除每两个字符串之间的关系, dp[ state ][ i ] 表示在state的状态下, 最后一个字符串是第i ...

随机推荐

  1. MySQL的字符串连接函数CONCAT, CONCAT_WS,GROUP_CONTACT

    本文转载自de.cel<MySQL的字符串连接函数CONCAT, CONCAT_WS,GROUP_CONCAT>   在搜索Mysql中怎么实现把一列的多行数据合并成一行时,找到了grou ...

  2. dsu on tree:关于一类无修改询问子树可合并问题

    dsu on tree:关于一类无修改询问子树可合并问题 开始学长讲课的时候听懂了但是后来忘掉了....最近又重新学了一遍 所谓\(dsu\ on\ tree\)就是处理本文标题:无修改询问子树可合并 ...

  3. foobar2000 iOS使用,并连接PC的歌曲进行播放

    foobar2000移动版下载地址:http://mobile.foobar2000.com/ 要实现歌曲互通有两种方法,使用iOS客户端的FTP Server用PC上传歌曲到iOS手机,和在PC上使 ...

  4. kis

    http://5.xp510.com:801/xp2011/%E9%87%91%E8%9D%B6kis%E4%B8%93%E4%B8%9A%E7%89%88.rar

  5. 报错: The type ByteInputStream is not accessible due to restriction on required library

    报错: Access restriction:The type JPEGCodec is not accessible due to restriction on required library C ...

  6. 如何用grep命令同时显示“匹配行”上下的n行?

    如何用grep命令同时显示匹配行上下的n行   标准unix/linux下的grep通过以下参数控制上下文 grep -C 5 foo file 显示file文件中匹配foo字串那行以及上下5行gre ...

  7. TUN/TAP区别

    在计算机网络中,TUN与TAP是操作系统内核中的虚拟网络设备.不同于普通靠硬件网路板卡实现的设备,这些虚拟的网络设备全部用软件实现,并向运行于操作系统上的软件提供与硬件的网络设备完全相同的功能. TA ...

  8. hdu 1879 继续畅通project

    本题链接:pid=1879http://">点击打开链接 本题大意: 输入n行数据.每行数据前两个表示该条路连通的两个村庄的编号,第三个表示修该条路的成本.最后的0或1表示该路未修或已 ...

  9. Android消息机制1-Handler(Java层)(转)

    转自:http://gityuan.com/2015/12/26/handler-message-framework/ 相关源码 framework/base/core/java/andorid/os ...

  10. Essay

    要养成先连续输入一对匹配的字符——比如"("和")",以及"{"和"}"——再在其中填写内容的习惯.如果先填写内容,很容 ...