[HNOI2006]最短母串问题 AC自动机
题面:洛谷
题解:
如果我们对这些小串建出AC自动机,那么我们所求的大串就是要求满足遍历过所有AC自动机上的叶子节点,且经过步数最少的串。如果有多个步数相同的串,要输出字典序最小的串。
在AC自动机上DP。
因为我们要求所求串内要出现所有给定小串,而小串个数较少,因此我们考虑状压,然后保存下val[i]表示走到这个点,就可以拥有哪些子串。因为一个非终止节点也可能包含给定小串,因此我们要在建好fail之后,继承一个点x的fail[x]所包含的小串,按编号大小更新即可保证在x被更新之前fail[x]已经被更新。
设f[i][j]表示在AC自动机的第i个节点上,状态为j的最小代价。
不知你是否注意到这个状态里没有限制长度和字典序?
其实这是因为我们可以用bfs解决问题,因为走任意一步的代价都是1,也就是每条边的权值都是1,所以如果我们用bfs的顺序来DP,那么谁先DP到拥有所有串,谁就是代价最小的方案。
那么怎么保证字典序最小?
我们只需要在bfs的时候从'a'开始枚举即可。
#include<bits/stdc++.h>
using namespace std;
#define R register int
#define AC 610
#define ac 4596
#define inf 2139062143 int n, m, maxn, tmp, head, tail;
int f[AC][ac], q1[AC * ac], q2[AC * ac];
short l1[AC][ac], l2[AC][ac];
int c[AC][], val[AC], fail[AC], go[AC], tot;
char s[AC]; inline void add()
{
int len = strlen(s + ), now = ;
for(R i = ; i <= len; i ++)
{
int v = s[i] - 'A';
if(!c[now][v]) c[now][v] = ++ tot, go[tot] = v;
now = c[now][v];
}
val[now] |= tmp, tmp <<= ;//还是要|= 的,否则要是有多个相同串在同一个节点结束就不好了,,,,
} #define q q1
void build()
{
for(R i = ; i < ; i ++)
if(c[][i]) q[++ tail] = c[][i];
while(head < tail)
{
int x = q[++ head];
for(R i = ; i < ; i ++)
{
if(c[x][i]) fail[c[x][i]] = c[fail[x]][i], q[++ tail] = c[x][i];
else c[x][i] = c[fail[x]][i];
}
}
for(R i = ; i <= tot; i ++) val[i] |= val[fail[i]];
}
#undef q void pre()
{
scanf("%d", &n);
maxn = ( << n) - , tmp = ;
memset(f, , sizeof(f));
for(R i = ; i <= n; i ++) scanf("%s", s + ), add();
} void bfs()
{
head = tail = ;
q1[++ tail] = , q2[tail] = ;
f[][] = ;
while(head < tail)
{
int x = q1[++ head], sta = q2[head];
if(sta == maxn)
{
head = ;
while(x)
{
q1[++ head] = go[x];
int tmp1 = x, tmp2 = sta;
x = l1[tmp1][tmp2], sta = l2[tmp1][tmp2];
}//因为要用2次,但是用了第一次之后x or sta就改变了,所以必须保存到临时变量
for(R i = head; i; i --) printf("%c", q1[i] + 'A');
return ;
}
for(R i = ; i < ; i ++)
{
int v = c[x][i], w = sta | val[v];
if(f[v][w] == inf) //bfs包括了最短和字典序最小
{
q1[++ tail] = v, q2[tail] = w;
f[v][w] = f[x][sta] + ;
l1[v][w] = x, l2[v][w] = sta;
}
}
}
} int main()
{
// freopen("in.in", "r", stdin);
pre();
build();
bfs();
// fclose(stdin);
return ;
}
[HNOI2006]最短母串问题 AC自动机的更多相关文章
- [HNOI2006]最短母串问题 --- AC自动机 + 隐式图搜索
[HNOI2006]最短母串问题 题目描述: 给定n个字符串(S1,S2.....,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,......,Sn)都是T的子串. 输入格式: 第 ...
- [BZOJ1195]:[HNOI2006]最短母串(AC自动机+BFS)
题目传送门 题目描述 给定n个字符串(S1,S2,…,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,…,Sn)都是T的子串. 输入格式 第一行是一个正整数n,表示给定的字符串的个数 ...
- [HNOI2006]最短母串问题——AC自动机+状压+bfs环形处理
Description 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T的子串. 32MB Input 第一行是一个正整数n(n< ...
- P2322 [HNOI2006]最短母串问题
P2322 [HNOI2006]最短母串问题 AC自动机+bfs 题目要求:在AC自动机建的Trie图上找到一条最短链,包含所有带结尾标记的点 因为n<12,所以我们可以用二进制保存状态:某个带 ...
- BZOJ_1195_[HNOI2006]最短母串_AC自动机+BFS+分层图
BZOJ_1195_[HNOI2006]最短母串_AC自动机+BFS+分层图 Description 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2, ...
- 【状态压缩dp】1195: [HNOI2006]最短母串
一个清晰的思路就是状压dp:不过也有AC自动机+BFS的做法 Description 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T ...
- 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 ...
随机推荐
- EmitMapper自动映射工具
在实体与DTO之间,我们一般都需要进行映射.如果手动的来进行转换,实在是太麻烦.所以就产生了很多映射工具,比如AutoMapper,EmitMapper.而经过一些对比,EmitMa ...
- php实现图形计算器
存档: index.php <html> <head> <title>图形计算器开发</title> <meta http-equiv=" ...
- power sequece
- linux中生成考核用的NTFS文件系统结构样例
实验NTFS-1说明:NTFS-1.img是一个包含NTFS文件系统的磁盘镜像,请使用winhex手工方式读出这个文件系统内的指定文件,并回答其md5 HASH值.要求: 1.利用WINHEX手工方式 ...
- Siki_Unity_2-7_Stealth秘密行动
Unity 2-7 Stealth秘密行动 Abstract:向量运算:Animation动画:Navigation寻路系统:Mecanim动画系统 任务1&2&3:游戏介绍 & ...
- Centos下安装并设置nginx开机自启动
一.在centos环境下安装下载并安装nginx,由于nginx需要依赖一些环境才能安装,主要依赖g++.gcc.openssl-devel.pcre-devel和zlib-devel这些环境,首先得 ...
- Spring Cloud(一):服务治理技术概览【Finchley 版】
Spring Cloud(一):服务治理技术概览[Finchley 版] 发表于 2018-04-14 | 更新于 2018-05-07 | Spring Cloud Netflix 是 Spr ...
- Kickstart 安装centos7
以前是怎么安装系统的 光盘(ISO文件,光盘的镜像文件)===>每一台物理机都得给一个光驱,如果用外置光驱的话,是不是每台机器都需要插一下 U盘:ISO镜像刻录到U盘==>需要每台机器都需 ...
- 基于NABCD评论作品,及改进建议
组名:杨老师粉丝群 组长:乔静玉 组员:吴奕瑶 刘佳瑞 公冶令鑫 杨磊 杨金铭 张宇 卢帝同 一.拉格朗日2018--<飞词> 1.1 NABCD分析 N(Need,需求) ...
- Android:有关菜单的学习(供自己参考)
Android:有关==菜单==的学习 上下文菜单 上下文菜单就是手机中对某一项进行==点击一定时间==后弹出的针对该项处理的菜单. context_menu.xml: <?xml versio ...