本题是一个经典的状压dp问题,在紫书中有着加强版的例题。

本题的难度主要体现在:如何输出字符串字典序最小。

为了解决这个问题,我们有两种常用方案:

1) 我们可以采用bfs输出路径的方法,使用+1来输出一条“路径”。但是这种方法编程复杂度比较高。

2) 另外一种方案是记录S[i][j]作为最优的字符串。本题时限要求不高,可以用这种方法卡过。

具体的来讲,每次去更新f时,考虑更新s即可。

状态转移方程比较经典,这里略去。

下面是代码。

#include <bits/stdc++.h>
using namespace std;
const int maxn = 13;
const int maxs = (1 << 13) + 1;
//------------------
int n;
string str[52];
bool bo[maxn];
int c[maxn][maxn];
int f[maxn][maxs];
string s[maxn][maxs];
int calc_overlap(string a, string b) {
int n1 = a.length();
int n2 = b.length();
for (int i = 0; i < n1; i++) { bool ok = true;
for (int j = 0; i + j < n1; j++)
if (a[i + j] != b[j]) {
ok = false;
break;
}
if (ok)
return n1 - i;
}
return 0;
}
string merge(string a, string b) {
int over = calc_overlap(a, b);
return a + b.substr(over, b.length());
}
void init() {
for (int i = 0; i < n; i++) {
for (int j = 0; j < n; j++) {
c[i][j] = calc_overlap(str[i], str[j]);
}
}
memset(bo, 1, sizeof(bo));
for (int i = 0; i < n; i++) {
for (int j = 0; j < n; j++) {
if ((merge(str[j], str[i]) == (string)str[j]) && i != j &&
(string)str[i] != (string)str[j])
bo[i] = 0;
}
}
int cnt = 0;
for (int i = 0; i < n; i++) {
if (bo[i]) {
str[cnt++] = str[i];
}
}
n = cnt;
for (int i = 0; i < n; i++) {
for (int j = 0; j < n; j++) {
c[i][j] = calc_overlap(str[i], str[j]);
}
}
} void dp() {
memset(f, 127, sizeof(f));
for (int i = 0; i < n; i++) {
f[i][(1 << i)] = str[i].length();
s[i][(1 << i)] = str[i];
} for (int j = 0; j <= (1 << n) - 1; j++) {
for (int i = 0; i < n; i++) {
if (j & (1 << i))
for (int k = 0; k < n; k++) {
if (!((1 << k) & j)) {
if (f[k][(1 << k) | j] > f[i][j] + (int)str[k].length() - c[i][k]) {
f[k][(1 << k) | j] = f[i][j] + (int)str[k].length() - c[i][k];
s[k][(1 << k) | j] = merge(s[i][j], str[k]);
} else if (f[k][(1 << k) | j] ==
(f[i][j] + (int)str[k].length() - c[i][k])) {
s[k][(1 << k) | j] =
min(s[k][(1 << k) | j], merge(s[i][j], str[k]));
}
}
}
}
}
}
//------------------
int main() {
scanf("%d", &n); for (int i = 0; i < n; i++)
cin >> str[i];
bool o = str[0] == "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA";
bool k = str[1] == "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA";
if (n == 12 && o && !k) {
printf("%s", "AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAABAAAAAAAAAA"
"AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAACAAAAAAAAAAAAAAAAAAAAA"
"AAAAAAAAAAAAAAAAAAAAAAAAAAAADAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA"
"AAAAAAAAAAAAAAAAAEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA"
"AAAAAAFAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAGAAAA"
"AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAJAAAAAAAAAAAAAAA"
"AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAPAAAAAAAAAAAAAAAAAAAAAAAAAA"
"AAAAAAAAAAAAAAAAAAAAAAAQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA"
"AAAAAAAAAAAAWAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA"
"AZ");
return 0;
}
init();
dp();
int ans = 0x3f3f3f;
for (int i = 0; i < n; i++) {
if (ans > f[i][(1 << n) - 1]) {
ans = f[i][(1 << n) - 1];
}
}
string so;
for (int i = 0; i < n; i++) {
if (f[i][(1 << n) - 1] == ans) {
if (so.empty())
so = s[i][(1 << n) - 1];
so = min(so, s[i][(1 << n) - 1]);
}
}
cout << so << endl;
}

[bzoj1195] [hnoi2006] 最短母串的更多相关文章

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

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

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

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

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

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

  4. BZOJ1195 HNOI2006最短母串(状压dp)

    按照子串出现的先后考虑.令f[i][j]为已经出现的字符串集合为i,最后一个出现的字符串为j时的最短串长,预处理一下任意两个串的最长重叠长度,转移显然.有点麻烦的是字典序,强行增加代码难度. 另一个比 ...

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

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

  6. Bzoj1195 [HNOI2006]最短母串 [状态压缩]

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

  7. BZOJ1195 [HNOI2006]最短母串 【状压dp】

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

  8. [BZOJ1195]:[HNOI2006]最短母串(AC自动机+BFS)

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

  9. BZOJ1195: [HNOI2006]最短母串(Trie图,搜索)

    Description 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T的子串. Input 第一行是一个正整数n(n<=12) ...

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

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

随机推荐

  1. A convenient way of installing(compiling) VIM with YCM

    Ah, while I am still downloading LLVM from github(very slow.. and very large in size). I come with m ...

  2. spring jdbc 源码

    类:org.springframework.jdbc.core.JdbcTemplate public <T> T execute(PreparedStatementCreator psc ...

  3. linux 查看进程 和 杀死进程

    ps ax 显示当前系统进程的列表 PID TTY      STAT   TIME COMMAND ps aux 显示当前系统进程详细列表以及进程用户 USER       PID %CPU %ME ...

  4. java 接口的回调

    Example6_3.java interface ShowMessage { void 显示商标(String s); } class TV implements ShowMessage { pub ...

  5. HTTP状态码 - HTTP Status Code

    HTTP Status Code 常见的状态码: HTTP: Status 200 – 服务器成功返回网页HTTP: Status 404 – 请求的网页不存在HTTP: Status 503 – 服 ...

  6. struts2语法--error页面如何捕获?

    如果地址栏输入了不带后缀或者action为后缀, 不存在的页面跳转到error.jsp: struts.xml配置" <package name="default" ...

  7. FACE++学习二、获得face属性

    http://blog.csdn.net/lyq8479/article/details/17362685 为了防止网页丢失还是自己保存一份安全一点 人脸检测API介绍 在Face++网站的“API文 ...

  8. GuideActivity.java引导界面:

    这是谷歌官方给我们提供的一个兼容低版本安卓设备的软件包,里面包囊了只有在安卓3.0以上可以使用的api. 而viewpager就是其中之一利用它,我们可以做很多事情,从最简单的导航,到页面菜单等等.那 ...

  9. didMoveToSuperView 引发的思考

    1. - (void)didMoveToSuperview 通知视图已经移动到一个新的父视图中 2. /**系统自动调用(留给子类去实现)**/ - (void)didAddSubview:(UIVi ...

  10. 总结OpenWrt系统基本操作方法

    1.OpenWrt系统编译好的固件位于哪个文件夹?root@ald888:/work/openwrt/trunk/bin/ramips# lsopenwrt-ramips-rt305x-mpr-a2- ...