题目大意:给定一个字符串集,求一个最短字串,使得该集合内的串都是该串的一个子串

算法:AC自动机+最短路+状压DP

注意空间限制

#include"cstdio"
#include"cstring"
#include"iostream"
#include"algorithm"
#include"bitset"
#include"string"
using namespace std; const int MAXN=13;
const int MAXM=55;
const int siz=26; int n,cnt,minn;
string ans;
short q[MAXN*MAXM*siz];
int ln[MAXN*MAXM],f[MAXN][MAXN],I[MAXN][1<<12];
string G[MAXN*MAXM],g[MAXN][MAXN],H[MAXN][1<<12];
int rid[MAXN];
char ch[MAXM*MAXN];
bitset<MAXN*MAXM> vis;
struct rpg{
int fail;
int sn[siz];
}T[MAXN*MAXM]; void ins(char *ch,int id)
{
int ln=strlen(ch+1),nw=0;
for(int i=1;i<=ln;++i){
if(!T[nw].sn[ch[i]-'A']) T[nw].sn[ch[i]-'A']=++cnt;
nw=T[nw].sn[ch[i]-'A'];
}rid[id]=nw;
return;
} void Getfail()
{
int hd=1,tl=0;
for(int i=0;i<siz;++i) if(T[0].sn[i]) q[++tl]=T[0].sn[i];
while(hd<=tl){
int nw=q[hd++];
for(int i=0;i<siz;++i){
if(T[nw].sn[i]){
T[T[nw].sn[i]].fail=T[T[nw].fail].sn[i];
q[++tl]=T[nw].sn[i];
}else T[nw].sn[i]=T[T[nw].fail].sn[i];
}
}return;
} void SPFA(int s)
{
int hd=1,tl=1;
memset(ln,0x7f,sizeof(ln));
for(int i=1;i<=cnt;++i) G[i].clear();
q[hd]=s;ln[s]=0;vis[s]=1;
while(hd<=tl){
int nw=q[hd++];vis[nw]=0;
if(ln[T[nw].fail]>ln[nw]){
ln[T[nw].fail]=ln[nw];
G[T[nw].fail]=G[nw];
if(!vis[T[nw].fail]){
vis[T[nw].fail]=1;
q[++tl]=T[nw].fail;
}
}for(int i=0;i<siz;++i){
string str=G[nw]+(char)(i+'A');
if(ln[T[nw].sn[i]]<ln[nw]+1) continue;
if(ln[T[nw].sn[i]]>ln[nw]+1){
ln[T[nw].sn[i]]=ln[nw]+1;
G[T[nw].sn[i]]=str;
if(!vis[T[nw].sn[i]]){
vis[T[nw].sn[i]]=1;
q[++tl]=T[nw].sn[i];
}
}if(G[T[nw].sn[i]]<=str) continue;
G[T[nw].sn[i]]=str;
if(!vis[T[nw].sn[i]]){
vis[T[nw].sn[i]]=1;
q[++tl]=T[nw].sn[i];
}
}
}return;
} void res()
{
SPFA(0);
for(int i=1;i<=n;++i){
f[0][i]=ln[rid[i]];
g[0][i]=G[rid[i]];
}for(int i=1;i<=n;++i){
SPFA(rid[i]);
for(int j=1;j<=n;++j){
f[i][j]=ln[rid[j]];
g[i][j]=G[rid[j]];
}
}return;
} void slv()
{
memset(I,0x7f,sizeof(I));
for(int i=1;i<=n;++i) I[i][1<<i-1]=f[0][i],H[i][1<<i-1]=g[0][i];
for(int i=1;i<1<<n;++i){
for(int j=1;j<=n;++j){
if((i>>j-1)&1==0) continue;
for(int k=1;k<=n;++k){
if((i>>k-1)&1) continue;
if(I[k][i|(1<<k-1)]<I[j][i]+f[j][k]) continue;
if(I[k][i|(1<<k-1)]>I[j][i]+f[j][k]){
I[k][i|(1<<k-1)]=I[j][i]+f[j][k];
H[k][i|(1<<k-1)]=H[j][i]+g[j][k];
}else if(H[k][i|(1<<k-1)]>H[j][i]+g[j][k]){
H[k][i|(1<<k-1)]=H[j][i]+g[j][k];
}
}
}
}ans=H[1][(1<<n)-1],minn=I[1][(1<<n)-1];
for(int i=2;i<=n;++i){
if(minn>I[i][(1<<n)-1]){
minn=I[i][(1<<n)-1];
ans=H[i][(1<<n)-1];
}else if(minn==I[i][(1<<n)-1]&&ans>H[i][(1<<n)-1]){
ans=H[i][(1<<n)-1];
}
}return;
} int main()
{
scanf("%d",&n);
for(int i=1;i<=n;++i) scanf("%s",ch+1),ins(ch,i);
Getfail();res();slv();
cout<<ans<<endl;
return 0;
}

[HNOI2006]最短母串问题的更多相关文章

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

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

  2. 2782: [HNOI2006]最短母串

    2782: [HNOI2006]最短母串 Time Limit: 1 Sec  Memory Limit: 128 MBSubmit: 3  Solved: 2[Submit][Status][Web ...

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

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

  4. P2322 [HNOI2006]最短母串问题

    P2322 [HNOI2006]最短母串问题 AC自动机+bfs 题目要求:在AC自动机建的Trie图上找到一条最短链,包含所有带结尾标记的点 因为n<12,所以我们可以用二进制保存状态:某个带 ...

  5. [HNOI2006]最短母串问题 --- AC自动机 + 隐式图搜索

    [HNOI2006]最短母串问题 题目描述: 给定n个字符串(S1,S2.....,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,......,Sn)都是T的子串. 输入格式: 第 ...

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

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

  7. BZOJ_1195_[HNOI2006]最短母串_AC自动机+BFS+分层图

    BZOJ_1195_[HNOI2006]最短母串_AC自动机+BFS+分层图 Description 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2, ...

  8. [bzoj1195][HNOI2006]最短母串_动态规划_状压dp

    最短母串 bzoj-1195 HNOI-2006 题目大意:给一个包含n个字符串的字符集,求一个字典序最小的字符串使得字符集中所有的串都是该串的子串. 注释:$1\le n\le 12$,$1\le ...

  9. BZOJ1195[HNOI2006]最短母串——AC自动机+BFS+状态压缩

    题目描述 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T的子串. 输入 第一行是一个正整数n(n<=12),表示给定的字符串的 ...

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

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

随机推荐

  1. Leetcode 102 二叉树的层次遍历 Python

    二叉树的层次遍历 给定一个二叉树,返回其按层次遍历的节点值. (即逐层地,从左到右访问所有节点). 例如: 给定二叉树: [3,9,20,null,null,15,7],   3   / \ 9 20 ...

  2. Set的总结

    Set最重要的操作是查找,为查找而设计.存入HashSet的元素必须定义hashCode(); Set不保存重复的元素,元素必须唯一.通过equals()方法一确保对象的唯一性. Set中最常被用于归 ...

  3. C#-集合及特殊集合——★★哈希表集合★★

    集合的基本信息: System.Collections命名空间包含接口和类,这些接口和类定义各种对象(如列表.队列.位组数.哈希表和字典)的集合. System.Collections.Generic ...

  4. 洛谷P3830 [SHOI2012]随机树(期望dp)

    题面 luogu 题解 第一问: 设\(f[i]\)表示\(i\)步操作后,平均深度期望 \(f[i] = \frac {f[i - 1] * (i - 1)+f[i-1]+2}{i}=f[i-1]+ ...

  5. Mac下使用Wine安装Notepad++

    下载: (链接: https://pan.baidu.com/s/1miOjLXY 密码: 2egg) 安装: 1.安装Wine 参考:http://www.cnblogs.com/EasonJim/ ...

  6. LINQ入门教程之各种标准查询操作符(二)

    续上篇LINQ入门教程之各种标准查询操作符(一) LINQ入门教程之各种标准查询操作符(二) 7.  聚合操作符 8.  集合操作符 9.  生成操作符 #region 生成操作符 即从现有序列的值中 ...

  7. php 判断字符串之间包含关系

    之前常用stristr ,  strpos判断. 因为处理1000W * 1000W级别,循环就是漫长漫长... 在此,对stristr, strpos, explode判断字符串包含关系处理速度对比 ...

  8. exe4j生成的exe反编译成java代码

    很早以前写了一个java串口小程序,现在只有exe4j打包后的源程序了,最近又要用,折腾了一下发现其实要找回来也很简单,这里记录一下,以免以后忘记. exe4j只是将java程序,使用自己的方式打包了 ...

  9. JavaScript设计模式-18.享元模式

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  10. IDEA 启动项目,tomcat中配置的虚拟路径无法使用

    有时候,使用idea启动项目,非动静分离项目,直接根据图片url地址显示图片,会发现图片无法显示,tomcat中配置的虚拟路径无法使用,这时候需要配置idea.选择路径,然后给与一个访问名就行了.