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),表示给定的字符串的 ...
 
随机推荐
- Java进阶知识点2:看不懂的代码 - 协变与逆变
			
一.背景 要搞懂Java中的协办与逆变,不得不从继承说起,如果没有继承,协变与逆变也天然不存在了. 我们知道,在Java的世界中,存在继承机制.比如MochaCoffee类是Coffee类的派生类,那 ...
 - 三、python沉淀之路--列表(list)
			
一.列表提供的一些方法 1.append():在原值最后追加 li = [11,22,33,44,55,] li.append(99) print(li) li.append('中国') print( ...
 - expect 切换用户
			
安装expect yum install expect -y #!/bin/expect -f spawn su - expect "Password:" send "n ...
 - Oracle中OEM的启动与关闭
			
我已经选择安装了,但安装后发现开始菜单里并没有OEM,在哪里可以打开呢? 从Oracle10g开始,Oracle极大的增强了OEM工具,并通过服务器端进行EM工具全面展现.在10g中,客户端可以不必安 ...
 - ORACLE删除用户的有的表的方法
			
首先我们查询oracle用户下的所有表 select * from all_tab_comments -- 查询所有用户的表,视图等select * from user_tab_comments ...
 - UEditor富文本编辑器的图片上传 http://fex.baidu.com/ueditor/#server-deploy
			
http://fex.baidu.com/ueditor/#server-deploy http://fex.baidu.com/ueditor/#server-path 首先 editor配置文件中 ...
 - DATAX动态参数数据传递
			
实例:ORACLE到ORACLE的数据传递 编写job.xml文件,添加变量参数 执行datax.py文件时记得带参数 格式:./datax.py –p"-Ddbname=*** -Di ...
 - Oracle 监听莫名死掉
			
有一台oracle 10g的监听莫名死掉,进行查看 select * from v$version Oracle Database 10g Enterprise Edition Release 10. ...
 - socket模型
			
Socket: "主机" + "端口" = 套接字/插座; 仅仅是一个通信模型,不属于七层协议(网络协议). 一台电脑(IP)的一个应用程序(端口) 和 另一台 ...
 - Java学习之系统高可用性渲染接口日志自动服务降级
			
背景:公司都追求系统的高可用性,这里不可用时间就是其中很重要的一个指标,为此在做系统功能升级迭代的过程中如何快速处理异常恢复正常功能极为重要.现在对新增模块的要求是都增加开关,方便快速关闭异常模块,但 ...