题面

问题描述

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

输入

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

输出

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

Sample Input

2
ABCD
BCDABC

Sample Output

ABCDABC

题解

AC自动机第一题...

实际上这题用到的是trie图, 状压DP即可. 每个节点记录在什么状态下被经过, 用于去重. 时间复杂度和空间复杂度都是\(2^nnL\)

#include <cstdio>
#include <cstring>
#include <deque> const int N = 12, LEN = 50;
int n;
char ans[N * LEN];
struct ACautomaton
{
struct node
{
node *suc[26], *fl;
int vst[1 << N], ed;
inline node()
{
for(int i = 0; i < 26; ++ i)
suc[i] = NULL;
memset(vst, 0, sizeof(vst));
ed = -1;
}
}*rt;
inline ACautomaton()
{
rt = new node;
rt->fl = rt;
}
inline void insert(char *str, int len, int id)
{
node *u = rt;
for(int i = 0; i < len; u = u->suc[str[i] - 'A'], ++ i)
if(u->suc[str[i] - 'A'] == NULL)
u->suc[str[i] - 'A'] = new node;
u->ed = id;
}
inline void build()
{
static std::deque<node*> que;
que.clear();
for(int i = 0; i < 26; ++ i)
if(rt->suc[i] != NULL)
rt->suc[i]->fl = rt, que.push_back(rt->suc[i]);
for(; ! que.empty(); que.pop_front())
{
node *u = que.front();
for(int i = 0; i < 26; ++ i)
if(u->suc[i] != NULL)
{
node *p = u->fl;
for(; p != rt && p->suc[i] == NULL; p = p->fl);
u->suc[i]->fl = p->suc[i] == NULL ? p : p->suc[i];
que.push_back(u->suc[i]);
}
for(int i = 0; i < 26; ++ i)
if(u->suc[i] == NULL)
u->suc[i] = u->fl->suc[i];
}
}
struct state
{
node *u;
int lst, rec, c;
inline state(node *_u, int _lst, int _rec, int _c)
{
u = _u, lst = _lst, rec = _rec, c = _c;
}
inline state() {}
};
inline void work()
{
static state que[(1 << N) * N * LEN];
int L = 0, R = 0;
que[R ++] = state(rt, -1, 0, -1);
for(; ; L ++)
{
state cur = que[L];
node *u = cur.u;
int rec = cur.rec;
if(~ u->ed)
rec |= 1 << u->ed, u->vst[rec] = 1;
if(rec == (1 << n) - 1)
break;
for(int i = 0; i < 26; ++ i)
if(u->suc[i] != NULL && ! u->suc[i]->vst[rec])
que[R ++] = state(u->suc[i], L, rec, i), u->suc[i]->vst[rec] = 1;
}
int len = 0;
for(; L; L = que[L].lst)
ans[len ++] = 'A' + que[L].c;
for(int i = len - 1; ~ i; -- i)
putchar(ans[i]);
}
}ACA;
int main()
{ #ifndef ONLINE_JUDGE
freopen("BZOJ1195.in", "r", stdin);
freopen("BZOJ1195.out", "w", stdout);
#endif scanf("%d\n", &n);
for(int i = 0; i < n; ++ i)
{
static char str[LEN];
scanf("%s", str);
ACA.insert(str, strlen(str), i);
}
ACA.build();
ACA.work();
}

HNOI 2006 BZOJ 1195 最短母串的更多相关文章

  1. [BZOJ 1195] 最短母串

    Link:https://www.lydsy.com/JudgeOnline/problem.php?id=1195 Solution: 看到数据范围n<=12,就要往状压DP上想 为了保证后项 ...

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

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

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

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

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

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

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

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

  6. 【loj10061】最短母串

    #10061. 「一本通 2.4 练习 4」最短母串 内存限制:512 MiB 时间限制:1000 ms 标准输入输出 题目类型:传统 评测方式:文本比较 上传者: 1bentong 提交    提交 ...

  7. [BZOJ1195]最短母串

    1195: [HNOI2006]最短母串 Time Limit: 10 Sec  Memory Limit: 32 MB Description 给定n个字符串(S1,S2,„,Sn),要求找到一个最 ...

  8. 2782: [HNOI2006]最短母串

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

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

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

随机推荐

  1. Android stadio

    Android stadio 最近遇到大问题,就是主功能行.但是让它做库工程,他就不管用. 但是在eclipse里面就可以.

  2. hdu3374 String Problem 最小最大表示法 最小循环节出现次数

    #include <iostream> #include <cstring> #include <cstdio> using namespace std; int ...

  3. 55、android app借助友盟实现微信授权登录

    一.去微信开放平台的管理中心申请移动设备的审核(需进行开发者资质认证,每年300元) 1.获取应用的签名 2.在微信开放平台申请移动应用 两个注意点:①签名要填对 ②应用的包名要写对(tips: co ...

  4. python2.7运行报警告:UnicodeWarning: Unicode equal comparison failed to convert both arguments to Unicode - interpreting them as being unequal解决办法

    1. 程序源代码报错部分: #选择年级if grade == '幼升小': outline.nianji().pop(0).click()elif grade == "一年级": ...

  5. python-os模块及md5加密

    常用内置方法 __doc__打印注释 __package__打印所在包 __cached__打印字节码 __name__当前为主模块是__name__ == __main__ __file__打印文件 ...

  6. mojoportal在IE10中点击ImageButton出错的处理方法

    在ie10中,如果点击了mojoportal中的imagebutton,会出现错误,在ie10之前的浏览器,及ie10的兼容模式中及谷歌浏览器中都不会出现. 日志中 错误信息如下: 2013-09-2 ...

  7. 频繁模式挖掘中Apriori、FP-Growth和Eclat算法的实现和对比(Python实现)

    最近上数据挖掘的课程,其中学习到了频繁模式挖掘这一章,这章介绍了三种算法,Apriori.FP-Growth和Eclat算法:由于对于不同的数据来说,这三种算法的表现不同,所以我们本次就对这三种算法在 ...

  8. 【翻译】Apache软件基金会1

    最近有点看不进去书,所以就找点东西翻译下,正好很想了解Apache基金会都有什么开源项目,每天找点事时间翻译翻译,还可以扩展下视野. 今天就看了两个,第一个是关于.NET的,不再兴趣范围内.第二个还挺 ...

  9. [LOJ#2327]「清华集训 2017」福若格斯

    [LOJ#2327]「清华集训 2017」福若格斯 试题描述 小d是4xx9小游戏高手. 有一天,小d发现了一个很经典的小游戏:跳青蛙. 游戏在一个 \(5\) 个格子的棋盘上进行.在游戏的一开始,最 ...

  10. [LOJ#2323]「清华集训 2017」小Y和地铁

    [LOJ#2323]「清华集训 2017」小Y和地铁 试题描述 小Y是一个爱好旅行的OIer.一天,她来到了一个新的城市.由于不熟悉那里的交通系统,她选择了坐地铁. 她发现每条地铁线路可以看成平面上的 ...