我有病吧……明明直接枚举是否匹配就可以非要写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. zookeeper原理浅析(二)

    参考:https://www.cnblogs.com/leocook/p/zk_1.html 代码:https://github.com/littlecarzz/zookeeper 1. 数据模型 1 ...

  2. Maven创建项目时出现Generating project in Interactive mode就一直卡住的解决方案

    使用maven命令在创建项目的时候出现 Generating project in Interactive mode 然后就一直卡住 网上搜做了很多解决方案 有说各种方案的,最后找到了一种.实验成功 ...

  3. HDOJ 5416 CRB and Tree DFS

    CRB and Tree Time Limit: 8000/4000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) Tot ...

  4. javascript是什么?

    JavaScript 是脚本语言 JavaScript 是一种轻量级的编程语言. JavaScript 是可插入 HTML 页面的编程代码. JavaScript 插入 HTML 页面后,可由全部的现 ...

  5. Windows——cmd findstr 字符串查找增强使用说明

    在文件中寻找字符串. 复制代码代码如下: FINDSTR [/B] [/E] [/L] [/R] [/S] [/I] [/X] [/V] [/N] [/M] [/O] [/P] [/F:file] [ ...

  6. crm使用soap删除下拉框

    //C# 代码: //DeleteOptionSetRequest request = new DeleteOptionSetRequest(); //request.Name = "new ...

  7. 关于Python中正则表达式的反斜杠问题

    之前总是搞不明白正则表达式中的反斜杠的问题.今天经过查阅资料终于搞明白了. 其中最重要的一点就是Python自己的字符串中定义的反斜杠也是转义字符,而正则表达式中的反斜杠也是转义字符,所以正则表达式中 ...

  8. 利用wxpython编写GUI

    首先建立一个简单的布局合理的图形界面,但是其中按下按键没有事情发生 #encoding=utf-8 __author__ = 'heng' #编写一个GUI import wx app = wx.Ap ...

  9. 块状元素的text-align对齐属性

    能够为块状元素(div,h1,h2,form等)内容设置位置text-align:center,left;right;

  10. Cleave js 使用

    1111111111111111 xxxxxx Cleave.js 键入时格式化< input />内容   信用卡号码格式 明确   美国运通:从34/37开始 34   签证:从4开始 ...