【loj10061】最短母串
题目描述
原题来自:HNOI 2006
给定 n个字符串 S1,S2,⋯,Sn 要求找到一个最短的字符串 T,使得这 n 个字符串都是 T 的子串。
输入格式
第一行是一个正整数 n,表示给定的字符串的个数;
以下的 n 行,每行有一个全由大写字母组成的字符串。
输出格式
只有一行,为找到的最短的字符串 T。
在保证最短的前提下,如果有多个字符串都满足要求,那么必须输出按字典序排列的第一个。
样例
样例输入
2
ABCD
BCDABC
样例输出
ABCDABC
数据范围与提示
对于全部数据,1≤n≤12,1≤∣Si∣≤50。
题意:(挂loj是因为我的解法在bzoj上T了并且我不想调QAQ)
给你$n$个字符串,求他们的最短母串,即求一个最短字符串使得这$n$个字符串都是它的子串。若长度相同输出字典序最小的。
题解:
建出$AC$自动机后问题转化成找一条最短的路径覆盖所有$end$节点。
那么我们可以考虑对每个节点维护一个压缩状态表示从根走到该点包含了哪些字符串。
然后从根开始沿$bfs$序遍历(保证字典序最小)并记录路径,若合法则输出答案。
这题主要是教会了我一个真理:
当你发现某个代码已经改不出来的时候,重构代码总是会给你惊喜……
代码:
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<queue> using namespace std;
#define MAXN 15
#define MAXM 605
#define ll long long struct node{
int son[],s;
}tree[MAXM];
struct way{
int las; char ch;
}p[MAXM*(<<MAXN)];
int tot=,nxt[MAXM];
bool vis[MAXM][<<MAXN];
char str[MAXN],ans[MAXN];
inline int read(){
int x=,f=;
char c=getchar();
for(;!isdigit(c);c=getchar())
if(c=='-')
f=-;
for(;isdigit(c);c=getchar())
x=x*+c-'';
return x*f;
} inline void add(int p,char *str){
int N=strlen(str),u=;
for(int i=;i<N;i++){
int ch=str[i]-'A';
if(!tree[u].son[ch])
tree[u].son[ch]=++tot;
u=tree[u].son[ch];
}
tree[u].s|=(<<p);
return;
} inline void get_nxt(){
for(int i=;i<;i++)
tree[].son[i]=;
queue<int> q; q.push();
while(!q.empty()){
int u=q.front(); q.pop();
for(int i=;i<;i++){
if(tree[u].son[i]){
q.push(tree[u].son[i]);
nxt[tree[u].son[i]]=tree[nxt[u]].son[i];
tree[tree[u].son[i]].s|=tree[tree[nxt[u]].son[i]].s;
}
else tree[u].son[i]=tree[nxt[u]].son[i];
}
}
return;
} inline void solve(int N){
int end=(<<N)-;
queue<int> q1,q2;
q1.push(); q2.push();
int hd=,tl=;
while(!q1.empty() && !q2.empty()){
int u=q1.front();q1.pop();
int sta=q2.front();q2.pop();
if(sta==end){
int cnt=;
for(;hd>;hd=p[hd].las)
ans[++cnt]=p[hd].ch;
for(int i=cnt;i>=;i--)
printf("%c",ans[i]);
printf("\n");
return;
}
for(int i=;i<;i++){
if(!vis[tree[u].son[i]][(tree[tree[u].son[i]].s)|(sta)]){
vis[tree[u].son[i]][(tree[tree[u].son[i]].s)|(sta)]=;
q1.push(tree[u].son[i]);
q2.push((tree[tree[u].son[i]].s)|(sta));
p[++tl].las=hd,p[tl].ch=i+'A';
}
}
hd++;
}
return;
} int main(){
int N=read();
for(int i=;i<N;i++)
scanf("%s",str),add(i,str);
get_nxt(); solve(N);
return ;
}
【loj10061】最短母串的更多相关文章
- 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 ...
- BZOJ1195[HNOI2006]最短母串——AC自动机+BFS+状态压缩
题目描述 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T的子串. 输入 第一行是一个正整数n(n<=12),表示给定的字符串的 ...
- 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,所以我们可以用二进制保存状态:某个带 ...
- [BZOJ1195]最短母串
1195: [HNOI2006]最短母串 Time Limit: 10 Sec Memory Limit: 32 MB Description 给定n个字符串(S1,S2,„,Sn),要求找到一个最 ...
- [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, ...
随机推荐
- MapReduce算法形式二:去重(HashSet)
案例二:去重(shuffle/HashSet等方法)shuffle主要针对的是key去重HashSet主要针对values去重
- Android IntentService的使用和源代码分析
引言 Service服务是Android四大组件之中的一个,在Android中有着举足重轻的作用.Service服务是工作的UI线程中,当你的应用须要下载一个文件或者播放音乐等长期处于后台工作而有没有 ...
- Struts%$#区别
1.#符号的用途一般有三种.1)访问非根对象属性,例如示例中的#session.msg表达式,由于Struts 2中值栈被视为根对象,所以访问其他非根对象时,需要加#前缀.实际上,#相当于Action ...
- B.大钉骑马走江湖
江湖是什么,对于在象棋界厮杀的大钉来说,江湖就是一个矩阵,他的目标,就是在江湖之中骑着马,从他的位置出发,走到终点. 当然,大钉的马也遵从中国象棋中的“马走日”的规则,而且在矩阵中,也会有一些障碍物, ...
- css3中animation的应用
1.css3 的相关属性: 相关代码: div { animation-name: myfirst; //动画的名称 animation-duration: 5s; //动画一个周期需要5秒 anim ...
- html的书写规范,有哪些注意点
1.最开始的声明格式与编码格式,注意html5与和html4.0的区别,注意对不同浏览器的渲染作用: 2.<head></head>标签中的相关内容的编写: 3.确保引入的jq ...
- MySQL数据库设计常犯的错以及对性能的影响
1.过分的反范式化为表建立太多的列 我们在设计数据库的结构时,比较容易犯的第一个错误就是对表进行了过分的反范式化的设计,这就容易造成了表中的列过多,虽然说Mysql允许为一个表建立很多的列,但是由于M ...
- hel软工网络16个人作业1
1Task1:注册个人博客账号 1Task2:注册码云账号 1Task3:提出问题 3.1问题一:软件工程是什么? 在第一章中我们可以从P8得到: 1.软件工程就是把系统的.有序的.可量化的方法应用到 ...
- 嵌入式Linux学习方法——给那些彷徨者(下)
上一章解决了嵌入式Linux的“学什么”问题,这一章则具体来说一下“怎么学”. 只要做好以下3点: 1.学习顺序.学习任何东西都应该由浅入深,不能一口吃下一个大胖子,得循序渐进.很多刚开始想学习Lin ...
- bzoj2337 XOR和路径——高斯消元
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2337 异或就一位一位考虑: x为到n的概率,解方程组即可: 考虑了n就各种蜜汁错误,所以索性 ...