Description

给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T的子串。

Input

第一行是一个正整数n(n<=12),表示给定的字符串的个数。
以下的n行,每行有一个全由大写字母组成的字符串。每个字符串的长度不超过50.

Output

只有一行,为找到的最短的字符串T。在保证最短的前提下,
如果有多个字符串都满足要求,那么必须输出按字典序排列的第一个。

Sample Input

2
ABCD
BCDABC

Sample Output

ABCDABC

拿这题练了练数组版的AC自动机

数据范围显然状压 插入时预处理出每个节点是哪个串的结束节点

然后建图 连fail指针的时候合并状态

为了最短显然需要按层转移,所以bfs

可以保证一达到末状态就return得到最优解

数组

#include<queue>
#include<cstdio>
#include<iostream>
#include<cstring>
#include<cstdlib>
using namespace std;
int n,tot=;
char s[];
struct trie
{
int son[],st,fail;
}t[];
void ins(char* str,int k)
{
int l=strlen(str+),root=;
for(int i=;i<=l;i++)
{
int x=str[i]-'A';
if(!t[root].son[x])t[root].son[x]=++tot;
root=t[root].son[x];
}
t[root].st|=<<(k-);
}
void build()
{
queue<int> q;
for(int i=;i<;i++)
if(t[].son[i])q.push(t[].son[i]);
while(!q.empty())
{
int x=q.front();
q.pop();
for(int i=;i<;i++)
{
int &y=t[x].son[i];
if(!y)
{
y=t[t[x].fail].son[i];
continue;
}
t[y].fail=t[t[x].fail].son[i];
t[y].st|=t[t[y].fail].st;
q.push(y);
}
}
}
bool v[][(<<)+];
int let[(<<)+],fa[(<<)+],ans[],num=;
void bfs()
{
queue<pair<int,int> > q;
q.push(make_pair(,));
int f=,ss=;
while(!q.empty())
{
int x=q.front().first,nowst=q.front().second;
q.pop();
if(nowst==(<<n)-)
{
for(int i=f;i;i=fa[i])
ans[++num]=let[i];
for(int i=num;i;i--)putchar(ans[i]+'A');
return ;
}
for(int i=;i<;i++)
{
int y=t[x].son[i],nxtst=nowst|t[y].st;
if(!v[y][nxtst])
{
v[y][nxtst]=;
ss++;
fa[ss]=f;
let[ss]=i;
q.push(make_pair(y,nxtst));
}
}
f++;
}
}
int main()
{
scanf("%d",&n);
for(int i=;i<=n;i++)
{
scanf("%s",s+);
ins(s,i);
}
build();
bfs();
return ;
}

指针

#include<cstdio>
#include<iostream>
#include<cstring>
#include<queue>
#include<cstdlib>
using namespace std;
const int N=;
int n;
char s[N];
struct node
{
node *son[];
int st;
bool vis[(<<)+];
node *fail;
node()
{
memset(this,,sizeof(node));
}
};
node *root;
void ini()
{
root=new node();
}
void ins(char *str,int num)
{
int l=strlen(str+);
node *now=root;
for(int i=;i<=l;i++)
{
if(!now->son[str[i]-'A'])now->son[str[i]-'A']=new node();
now=now->son[str[i]-'A'];
}
now->st|=<<(num-);
} void build()
{
queue<node*>q;
for(int i=;i<;i++)
{
if(root->son[i])
{
root->son[i]->fail=root;
q.push(root->son[i]);
}
else root->son[i]=root;
}
while(!q.empty())
{
node *x=q.front();
q.pop();
for(int i=;i<;i++)
{
if(x->son[i])
{
x->son[i]->fail=x->fail->son[i];
if(x->son[i]->fail)x->son[i]->st|=x->son[i]->fail->st;
q.push(x->son[i]);
}
else x->son[i]=x->fail->son[i];
}
}
}
int fa[],ans[],tot=,qwq[];
void bfs()
{
queue<node*> q;
queue<int> ST;
q.push(root);ST.push();
root->vis[]=;
int f=,ss=;
while(!q.empty())
{
node *x=q.front();int nowSt=ST.front();
q.pop();ST.pop();
// cout<<(x->st)<<endl;
if(nowSt==(<<n)-)
{
for(int i=f;i;i=fa[i])ans[++tot]=qwq[i];
for(int i=tot;i;i--)putchar(ans[i]+'A');
return ;
}
for(int i=;i<;i++)
{
if(!x->son[i])continue;
int state=nowSt|(x->son[i]->st);
if(!x->son[i]->vis[state])
{
x->son[i]->vis[state]=;
ss++;
fa[ss]=f;
qwq[ss]=i;
q.push(x->son[i]);
ST.push(state);
}
}
f++;
}
}
int main()
{
scanf("%d",&n);
ini();
for(int i=;i<=n;i++)
{
scanf("%s",s+);
ins(s,i);
}
build();
bfs();
return ;
}

(话说这题数组大小神TM坑啊……)

[HNOI2006]最短母串 (AC自动机+状压)的更多相关文章

  1. bzoj1195 [HNOI2006]最短母串 AC 自动机+状压+bfs

    题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=1195 题解 建立 AC 自动机,然后构建出 trie 图. 然后直接在 trie 图上走.但是 ...

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

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

  3. 【bzoj1195】[HNOI2006]最短母串 AC自动机+状态压缩+BFS最短路

    原文地址:http://www.cnblogs.com/GXZlegend/p/6825226.html 题目描述 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串 ...

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

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

  5. Bzoj1195 [HNOI2006]最短母串 [AC自动机]

    Time Limit: 10 Sec  Memory Limit: 32 MBSubmit: 1304  Solved: 439 Description 给定n个字符串(S1,S2,„,Sn),要求找 ...

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

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

  7. BZOJ1195 [HNOI2006]最短母串 AC自动机 bfs

    欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 传送门 - BZOJ1195 题意概括 给出一堆串,然后求一个包含这些串的所有串的最短的中的字典序最小的. 题解 先造一个AC ...

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

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

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

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

随机推荐

  1. 菜鸟的mongoDB学习---(五)MongoDB的limit、skip、sort方法

    limit方法 假设你须要在MongoDB中读取指定数量的数据记录.能够使用MongoDB的Limit方法,limit()方法接受一个数字參数,该參数指定从MongoDB中读取的记录条数. mongo ...

  2. Eclipse Android环境配置

    1.离线安装ADT插件,先将ZIP包下载 Help- Install New Software- Add 重启 2.WIndows -Preference设置SDK目录

  3. Vijos 1523 贪吃的九头龙 【树形DP】

    贪吃的九头龙 背景 安徽省芜湖市第二十七中学测试题 NOI 2002 贪吃的九头龙(dragon) Description:OfficialData:OfficialProgram:Converted ...

  4. Struts数据验证

    Action类继承了ActionSupport类,而该类实现了Action.Validateable.ValidationAware.TextProvider.LocaleProvider和Seria ...

  5. bzoj 1046: [HAOI2007]上升序列【dp+二分】

    先从后到前做一个最长下降子序列的dp,记录f[i],我这里用的是二分(其实树状数组比较显然) 然后对于询问,超出最长上升子序列的直接输出:否则从前到后扫,f[i]>=x&&a[i ...

  6. bzoj [Usaco2010 Hol]cowpol 奶牛政坛【树链剖分】

    意识流虚树 首先考虑只有一个党派,那么可以O(n)求树的直径,步骤是随便指定一个根然后找距离根最远点,然后再找距离这个最远点最远的点,那么最远点和距离这个最远点最远的点之间的距离就是直径 那么考虑多党 ...

  7. 一个简陋的个人小项目,也是个人第一个真正意义上的独立项目——Graph

    由来 我最早接触到图这个概念是在大二的离散数学当中图论相关的内容,当时是以著名的哥尼斯堡七桥问题引出图论的概念,现在依然记忆犹新(不过只是记得这个名字,具体的解题思路我重新温习了一下才想起来),当时也 ...

  8. extjs grid禁止表格头部使用鼠标拖拽改变宽度

    extjs6 经典版 表格头部使用鼠标拖动 禁止改变列的宽度 只需要给grid 设置属性enableColumnResize:false就可以啦 xtype:'grid', enableColumnR ...

  9. 通过路由器的IP映射来解决,两个不同IP地址的PC机之间的从LAN口到WAN口的单向通讯问题

    1.问题假设: 在B机中IP地址与子网掩码都固定,网关是路由器的LAN口的IP地址,我们希望通过路由器来实现B机与A机之间的单向通讯问题,也就是说B可以ping通A且可以访问A提供的FTP站点. 2. ...

  10. javascript面试题集

    1.如何把一句英文每个单词第一个字母大写? var str = "what fuck is 1235 going on ?"; var newArr = str.split(&qu ...