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),表示给定的字符串的 ...
随机推荐
- 马拉车——模版+KMP——模版
void Manacher(){ ;t[i];++i,len+=){ s[i<<]='#'; |]=t[i]-'A'+'a'; |]=t[i]; } s[len++]='#'; ,pos= ...
- boost开发指南
C++确实很复杂,神一样的0x不知道能否使C++变得纯粹和干爽? boost很复杂,感觉某些地方有过度设计和太过于就事论事的嫌疑,对实际开发工作的考虑太过于理想化.学习boost本身就是一个复杂度,有 ...
- mysql索引攻略
本设计和优化专题转自博客园的Mysql的设计和优化专题 Explain优化查询检测 所谓索引就是为特定的mysql字段进行一些特定的算法排序,比如二叉树的算法和哈希算法,哈希算法是通过建立特征值,然后 ...
- hdu 1937 Finding Seats
Time Limit: 5000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submission( ...
- BZOJ5319: [Jsoi2018]军训列队
BZOJ5319: [Jsoi2018]军训列队 https://lydsy.com/JudgeOnline/problem.php?id=5319 分析: 易知把所有人按原本的顺序放到\([K,K+ ...
- 【java规则引擎】简单规则的rete网络示意图
一个Fact通过Session添加到规则网络中,如何进行规则匹配的大致过程如下 (1)通过根结点对象从EntryPointNode的Map集合中找到相应的EntryPointNode对象 (2)Ent ...
- Unity 头发随动效果
目标 实现角色的衣袖.头发.裙摆.披风.尾巴等,在角色运动时,可以产生随动的效果.类似王者荣耀角色展示界面. 准备 源码出出处:https://github.com/unity3d-jp/unityc ...
- java中初始化方法
本文主要是讲从<java编程思想>中看到的东西,是第七章复用类的开头内容,主要是类初始化的几种方法的位置,主要包括 1.在定义对象的地方.这意味着他们总是在构造器被调用前被初始化. 2.在 ...
- CUDA Pro Tip: Optimized Filtering with Warp-Aggregated Atomics
In this post, I’ll introduce warp-aggregated atomics, a useful technique to improve performance when ...
- Java基础--阻塞队列ArrayBlockingQueue
ArrayBlockingQueue是阻塞队列的一种,基于数组实现,长度固定,队尾添加,队首获取, 构造函数: ArrayBlockingQueue(int capacity) ArrayBlocki ...