我有病吧……明明直接枚举是否匹配就可以非要写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. Eventquery.vbs

    https://docs.microsoft.com/en-us/previous-versions/orphan-topics/ws.10/cc772995(v=ws.10)

  2. python学习之-- redis模块操作 HASH

    redis 操作 之 -Hash Hash 操作:hash在内存中的存储格式 name hash n1 ------> k1 -> v1 k2 -> v2 k3 -> v3hs ...

  3. B. Restaurant--cf579B (贪心)

    http://codeforces.com/problemset/problem/597/B 把右节点从小到大排序  在跑一遍就行了 #include <iostream> #includ ...

  4. 七天从零基础学习android(1)--配置环境

    在写这篇文的时候android开发经验还是0,是一个萌新,这是一篇记录一个萌新从零android编程基础到能编写一个记账本的开发过程(至少我是这样美好的希望着的) 首先因为是没有开发基础的,直接上百度 ...

  5. java基础 4 继承(1)抽象类与接口的区别

    抽象类: 是用来捕捉子类的通用特性的,至少包含一个抽象方法,该抽象方法必须在子类中实现,由于抽象类没有抽象方法的具体实现,因此不能对抽象类进行实例化. 接口: 定义了一组方法,是抽象方法的集合,但是接 ...

  6. Java 函数式接口 lambda

    import java.io.Serializable; import java.util.ArrayList; import java.util.List; public class Demo1 { ...

  7. Oracle冷备和热备脚本

    Oracle冷备和热备脚本 冷备脚本: set feedback off set heading off set verify  off set trimspool off set echo off ...

  8. O2O助汪峰成功逆袭,汪峰最终上头条了

    8月2日七夕情人节,汪峰<峰暴来临>演唱会在鸟巢10万人体育场唱响,各大报纸.站点娱乐板块并没有等来汪峰向国际章求婚的"头条",只是,与乐视合作现场演出+付费直播的O2 ...

  9. Linux中查看文件或者文件夹大小

    df -l 查看磁盘空间大小命令 df -hl  查看磁盘剩余空间 df -h  查看每个根路径的分区大小 du -sh  当前文件夹下所有文件大小(包括子文件大小 du -sm  [文件夹] 返回该 ...

  10. LoadRunner---http请求中对中文参数的处理

    Loadrunner 做保险承保业务测试 1. 保险正常业务流程:保费计算--->保存--->申请核保--->核保--->缴费(出保单) 问题一描述 保费计算接口中,需要把车牌 ...