唔,太菜了,弄了好几个小时。

状压dp,设$f_{s, i}$表示选了集合$s$,以$i$结尾的最短长度,设$g_{i, j}$表示串$i$的后缀和串$j$的前缀的最长匹配长度。

$f_{s, i} + len_{j} - g_{i, j} $ 可以转移到$f_{s | (1 << (j - 1)), j}$   $(i\in s, j\notin s)$。

转移的时候发现两个串的长度一样要把这两个的答案都弄出来比一比字典序。

如果一个串是另一个串的字串,不参与转移。

注意特判全部串相同的情况。

可以用kmp优化$g$的计算和判断子串,但是$n$太小了,所以时间并不会差太多。

时间复杂度$O(能过)$。

用AC自动机 + BFS转移会更优美.

Code:

#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std; const int N = ;
const int S = ( << ) + ;
const int L = ;
const int inf = 0x3f3f3f3f; int n, nowLen, len[N], g[N][N], f[S][N], pre[S][N];
bool mark[N]; struct MyStr {
char s[L];
} a[N]; inline void chkMin(int &x, int y) {
if(y < x) x = y;
} inline bool chk(int x, int y, int l) {
for(int i = len[x] - l + , j = ; j <= l; i++, j++)
if(a[x].s[i] != a[y].s[j]) return ;
return ;
} inline bool con(char *s1, int y) {
char *s2 = a[y].s;
if(len[y] > nowLen) return ;
for(int i = ; i <= nowLen; i++)
if(s1[i] == s2[]) {
bool flag = ;
for(int k = i + , j = ; j <= len[y]; j++, k++)
if(s2[j] != s1[k]) {
flag = ;
break;
}
if(flag) return ;
}
return ;
} inline void getS(int nowS, int now, char *str) {
if(!pre[nowS][now]) {
nowLen = ;
for(int i = ; i <= len[now]; i++)
str[++nowLen] = a[now].s[i];
return;
}
if(pre[nowS][now]) getS(nowS ^ ( << (now - )), pre[nowS][now], str);
// if(con(str, now)) return;
for(int i = g[pre[nowS][now]][now] + ; i <= len[now]; i++)
str[++nowLen] = a[now].s[i];
} inline bool myCmp(char *s1, char *s2, int l) {
for(int i = ; i <= l; i++)
if(s1[i] < s2[i]) return ;
else if(s1[i] > s2[i]) return ;
return ;
} int main() {
scanf("%d", &n);
for(int i = ; i <= n; i++) scanf("%s", a[i].s + );
// sort(a + 1, a + 1 + n, cmp);
for(int i = ; i <= n; i++) len[i] = strlen(a[i].s + ); for(int i = ; i <= n; i++)
for(int j = ; j <= n; j++) {
if(i == j) {
g[i][j] = len[i];
continue;
}
for(int k = min(len[i], len[j]); k >= ; k--)
if(chk(i, j, k)) {
g[i][j] = k;
break;
}
} for(int i = ; i <= n; i++) {
nowLen = len[i];
for(int j = ; j <= n; j++) {
if(j == i) continue;
if(con(a[i].s, j)) mark[j] = ;
}
} bool flag = ;
for(int i = ; i <= n; i++)
if(!mark[i]) {
flag = ;
break;
}
if(!flag) {
puts(a[].s + );
return ;
} /* printf("\n");
for(int i = 1; i <= n; i++)
printf("%s\n", a[i].s + 1);
printf("\n");
for(int i = 1; i <= n; i++, printf("\n"))
for(int j = 1; j <= n; j++)
printf("%d ", g[i][j]); */ memset(f, 0x3f, sizeof(f));
for(int i = ; i <= n; i++)
if(!mark[i]) f[ << (i - )][i] = len[i]; char str1[N * L], str2[N * L];
for(int s = ; s < ( << n); s++)
for(int i = ; i <= n; i++)
if((s & ( << (i - ))) && f[s][i] != inf && !mark[i])
for(int j = ; j <= n; j++)
if(!(s & ( << (j - ))) && !mark[j]) {
// getS(s, i, str1);
int now = len[j] - g[i][j];
if(f[s][i] + now < f[s | ( << (j - ))][j]) {
f[s| ( << (j - ))][j] = now + f[s][i];
pre[s| ( << (j - ))][j] = i;
} else if(f[s][i] + now == f[s | ( << (j - ))][j]) {
getS(s | ( << (j - )), j, str1), getS(s, i, str2);
for(int k = g[i][j] + ; k <= len[j]; k++)
str2[++nowLen] = a[j].s[k]; if(myCmp(str2, str1, nowLen)) pre[s | ( << (j - ))][j] = i;
}
} int ans = inf, curS = ( << n) - ;
for(int i = ; i <= n; i++)
if(mark[i]) curS ^= ( << (i - ));
for(int i = ; i <= n; i++)
chkMin(ans, f[curS][i]); int pos = ;
for(int i = ; i <= n; i++)
if(f[curS][i] == ans) {
getS(curS, i, str1);
pos = i;
break;
} for(int i = pos + ; i <= n; i++) {
if(f[curS][i] > ans) continue;
getS(curS, i, str2); /* for(int j = 1; j <= ans; j++) putchar(str1[j]); printf("\n");
for(int j = 1; j <= ans; j++) putchar(str2[j]); printf("\n"); */ if(myCmp(str2, str1, ans))
for(int j = ; j <= ans; j++) str1[j] = str2[j];
} for(int i = ; i <= ans; i++)
putchar(str1[i]);
printf("\n");
return ;
}

Luogu 2322 [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),表示给定的字符串的 ...

随机推荐

  1. C++中关于strtok()函数的用法

    strtok: #include <string.h> char *strtok(char *str, const char *delim); char *strtok_r(char *s ...

  2. Docker-安装与部署

    本文在CentsOS下安装Docker 1.安装前准备工作 系统要求: 在CentOS下需要64位的CentsOS 7  OS requirements To install Docker, you ...

  3. C#面向对象(一):明确几个简单的概念作为开胃菜

    绝对干货,总结了下C#面向对象的干货内容以及我几年来的使用心得,分享出来 先明确几个简单概念作为开胃菜: 1.命名空间 namespace BackStageManagement.Services 关 ...

  4. [Luogu2371][国家集训队]墨墨的等式

    luogu 题意 给出\(n,a_i,B_{min},B_{max}\),求使得\(a_1x_1+a_2x_2+...+a_nx_n=B\)存在一组非负整数解的\(B\in[B_{min},B_{ma ...

  5. verilog学习五点经验分享

    1.规范很重要工作过的朋友肯定知道,公司里是很强调规范的,特别是对于大的设计(无论软件还是硬件),不按照规范走几乎是不可实现的.逻辑设计也是这样:如果不按规范做的话,过一个月后调试时发现有错,回头再看 ...

  6. 手把手教你创建Azure ARM Template

    Azure的ARM模式在中国已经落地了.在ARM模式中,通过ARM的Template批量的创建各种资源是与ASM模式的最大的区别之一.目前Azure ARM的Template数量已经越来越多,更多的客 ...

  7. Jmeter & TICK

    背景:   本来只是想在将Jmeter的测试结果写入InfluxDB, 但发现从InfluxDB V1.3后开始, 已经不支持Web Admin interface, 才发现InfluxData 搞了 ...

  8. [转] Linux 查找文件内容

    Linux查找文件内容的常用命令方法. 从文件内容查找匹配指定字符串的行: $ grep "被查找的字符串" 文件名例子:在当前目录里第一级文件夹中寻找包含指定字符串的.in文件g ...

  9. Java-API-Package:javax.http.servlet

    ylbtech-Java-API-Package:javax.http.servlet 1.返回顶部 1. Package javax.servlet.http This chapter descri ...

  10. Ueditor/自定义配置

    UEditor除 了具有轻量.可定制等优点外,还始终将优化编辑操作.提升用户体验摆在了很重要的位置.在这一点上,除了对编辑器功能.性能.实现细节等不断地改进和追求 创新之外,众多灵活而人性化的自定义配 ...