Luogu 2322 [HNOI2006]最短母串问题
唔,太菜了,弄了好几个小时。
状压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]最短母串问题的更多相关文章
- bzoj 1195: [HNOI2006]最短母串 爆搜
1195: [HNOI2006]最短母串 Time Limit: 10 Sec Memory Limit: 32 MBSubmit: 894 Solved: 288[Submit][Status] ...
- 2782: [HNOI2006]最短母串
2782: [HNOI2006]最短母串 Time Limit: 1 Sec Memory Limit: 128 MBSubmit: 3 Solved: 2[Submit][Status][Web ...
- BZOJ 1195: [HNOI2006]最短母串
1195: [HNOI2006]最短母串 Time Limit: 10 Sec Memory Limit: 32 MBSubmit: 1346 Solved: 450[Submit][Status ...
- P2322 [HNOI2006]最短母串问题
P2322 [HNOI2006]最短母串问题 AC自动机+bfs 题目要求:在AC自动机建的Trie图上找到一条最短链,包含所有带结尾标记的点 因为n<12,所以我们可以用二进制保存状态:某个带 ...
- [HNOI2006]最短母串问题 --- AC自动机 + 隐式图搜索
[HNOI2006]最短母串问题 题目描述: 给定n个字符串(S1,S2.....,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,......,Sn)都是T的子串. 输入格式: 第 ...
- 【状态压缩dp】1195: [HNOI2006]最短母串
一个清晰的思路就是状压dp:不过也有AC自动机+BFS的做法 Description 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T ...
- BZOJ_1195_[HNOI2006]最短母串_AC自动机+BFS+分层图
BZOJ_1195_[HNOI2006]最短母串_AC自动机+BFS+分层图 Description 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2, ...
- [bzoj1195][HNOI2006]最短母串_动态规划_状压dp
最短母串 bzoj-1195 HNOI-2006 题目大意:给一个包含n个字符串的字符集,求一个字典序最小的字符串使得字符集中所有的串都是该串的子串. 注释:$1\le n\le 12$,$1\le ...
- BZOJ1195[HNOI2006]最短母串——AC自动机+BFS+状态压缩
题目描述 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T的子串. 输入 第一行是一个正整数n(n<=12),表示给定的字符串的 ...
随机推荐
- LKDBHelper Sqlite操作数据库
首先这里要说明一下,为什么用FMDB而不用Core Data呢,因为我们不知道Core Data是如何映射对象里面的属性关系的,如果我们更改了属性的话,就会报错 首先是创建LKDBHelper对象 L ...
- PHP学习创建水印,缩略图
今天网上学习了一段PHP创建缩略图还有打水印的代码,如下: 其中将图片的路径作为参数传给函数,打水印的过程就是首先获取图片和logo的参数信息,然后将logo图片拷贝到原图的某个位置,然后保存,水印打 ...
- Oracle中OEM的启动与关闭
我已经选择安装了,但安装后发现开始菜单里并没有OEM,在哪里可以打开呢? 从Oracle10g开始,Oracle极大的增强了OEM工具,并通过服务器端进行EM工具全面展现.在10g中,客户端可以不必安 ...
- tyvj 2054 [Nescafé29]四叶草魔杖——最小生成树+状压dp
题目:http://www.joyoi.cn/problem/tyvj-2054 枚举点集,如果其和为0,则作为一个独立的块求一下最小生成树.因为它可以不和别的块连边. 然后状压dp即可. 别忘了判断 ...
- C# 保护进程不被结束(源代码)防任务管理器结束进程
C# 保护进程不被结束(源代码)防任务管理器结束进程 Posted on 2013-03-25 16:03 快乐家++ 阅读(3173) 评论(3) 编辑 收藏 闲来无事,英语又学的太痛苦.看到我妈妈 ...
- IEEE1588精密网络同步协议(PTP)
1 引言 以太网技术由于其开放性好.价格低廉和使用方便等特点,已经广泛应用于电信级别的网络中,以太网的数据传输速度也从早期的10M提高到100M,GE,10GE.40GE,100GE正式产品也于20 ...
- mysql 自增id
在开发的时候遇到了 自增id变成2147483647 莫名其妙 然后发现是自己没把自增id改为 无符号的原因 把无符号勾上就ok了
- GUI练习中
总结:JFrame和Frame是有很大差别的. 不要混淆.否则方法是不能成功调用的 特别是背景色:JFrame.对象f在main里无法调用背景色前景色都不想显示 一下是书上的一段代码,编译错误,但是可 ...
- canvas绘制简单图形
canvas绘图篇: canvas绘制矩形: <!DOCTYPE html> <html> <head lang="en"> <meta ...
- spring事务 异常回滚
spring事务回滚 可抛出自定义继承自RuntimeException