Time Limit: 10 Sec  Memory Limit: 32 MB
Submit: 1304  Solved: 439

Description

给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T的子串。

Input

第一行是一个正整数n(n<=12),表示给定的字符串的个数。以下的n行,每行有一个全由大写字母组成的字符串。每个字符串的长度不超过50.

Output

只有一行,为找到的最短的字符串T。在保证最短的前提下,如果有多个字符串都满足要求,那么必须输出按字典序排列的第一个。

Sample Input

2
ABCD
BCDABC

Sample Output

ABCDABC

HINT

 

Source

AC自动机+BFS:http://www.cnblogs.com/SilverNebula/p/6445516.html

状压DP+神(bao)奇(li)预处理

先找出所有被其他串包含的串并扔掉(显然)

暴力预处理出每两个串连接起来,公共部分的长度。

进行状压DP,暴力储存每个状态对应的字符串。

在所有的11111...的最终状态中,暴力找出字典序最小的那个

是不是很神(bao)奇(li)?

不知道为何,用strcmp会TLE(也可能是别的处理不到位),各处优化了一下,手写了比较函数,终于过了

 #include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<queue>
using namespace std;
const int INF=0x3f3f3f3f;
const int mxn=;
int n;
struct bind{
char s[];
int len;
bool operator < (const bind y) const {
if(len!=y.len)return len<y.len;
for(int i=;i<len;i++)
if(s[i]!=y.s[i])return s[i]<y.s[i];
return ;
}
}f[<<][],s[];
int c[][];
bool ban[mxn]; bool ovl(int i,int j){//判断能否覆盖
if(s[i].len<s[j].len)return ;
char *p=strstr(s[i].s,s[j].s);
if(p==NULL)return ;
return ;
}
int clc(int x,int y){//统计共用串长度
bool flag=;
for(int i=max(,s[x].len-s[y].len);i<s[x].len;i++){
flag=;
for(int j=i;j<s[x].len;j++)
if(s[x].s[j]!=s[y].s[j-i]){flag=;break;}
if(flag)return s[x].len-i;//--1
}
return ;
}
bind merge(int S,int u,int v){//字符串合并
bind tmp=f[S][u];
strcat(tmp.s,s[v].s+c[u][v]);
tmp.len=f[S][u].len-c[u][v]+s[v].len;
// printf("merge:%d %d %d :%d\n",S,u,v,tmp.len);
return tmp;
}
void Dp(){
int i,j,ed=(<<n)-;
for(i=;i<=ed;i++)
for(j=;j<n;j++)f[i][j].len=INF;//init
for(i=;i<n;i++)f[<<i][i]=s[i];
for(i=;i<=ed;i++){
for(j=;j<n;j++){
if((i>>j)&)
for(int k=;k<n;k++){
if((i>>k)&)continue;
bind tmp=merge(i,j,k);
if(tmp<f[i|(<<k)][k])f[i|(<<k)][k]=tmp;
}
}
}
}
int main(){
int i,j;
scanf("%d",&n);
for(i=;i<n;i++)scanf("%s",s[i].s),s[i].len=strlen(s[i].s);
for(i=;i<n;i++)
for(j=;j<n;j++)
if(i!=j && ovl(i,j) && !ban[i])ban[j]=;
int cnt=;
for(i=;i<n;i++)if(!ban[i])s[cnt++]=s[i];
n=cnt;
for(i=;i<n;i++)
for(j=;j<n;j++)
if(i!=j)c[i][j]=clc(i,j);
Dp();
int ans=,ed=(<<n)-;
for(i=;i<n;i++)
if(f[ed][i]<f[ed][ans])ans=i;
printf("%s",f[ed][ans].s);
return ;
}

Bzoj1195 [HNOI2006]最短母串 [状态压缩]的更多相关文章

  1. BZOJ1195[HNOI2006]最短母串——AC自动机+BFS+状态压缩

    题目描述 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T的子串. 输入 第一行是一个正整数n(n<=12),表示给定的字符串的 ...

  2. [bzoj1195][HNOI2006]最短母串_动态规划_状压dp

    最短母串 bzoj-1195 HNOI-2006 题目大意:给一个包含n个字符串的字符集,求一个字典序最小的字符串使得字符集中所有的串都是该串的子串. 注释:$1\le n\le 12$,$1\le ...

  3. BZOJ1195 [HNOI2006]最短母串 AC自动机 bfs

    欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 传送门 - BZOJ1195 题意概括 给出一堆串,然后求一个包含这些串的所有串的最短的中的字典序最小的. 题解 先造一个AC ...

  4. [BZOJ1195]:[HNOI2006]最短母串(AC自动机+BFS)

    题目传送门 题目描述 给定n个字符串(S1,S2,…,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,…,Sn)都是T的子串. 输入格式 第一行是一个正整数n,表示给定的字符串的个数 ...

  5. Bzoj1195 [HNOI2006]最短母串 [AC自动机]

    Time Limit: 10 Sec  Memory Limit: 32 MBSubmit: 1304  Solved: 439 Description 给定n个字符串(S1,S2,„,Sn),要求找 ...

  6. BZOJ1195 HNOI2006最短母串(状压dp)

    按照子串出现的先后考虑.令f[i][j]为已经出现的字符串集合为i,最后一个出现的字符串为j时的最短串长,预处理一下任意两个串的最长重叠长度,转移显然.有点麻烦的是字典序,强行增加代码难度. 另一个比 ...

  7. BZOJ1195 [HNOI2006]最短母串 【状压dp】

    题目 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T的子串. 输入格式 第一行是一个正整数n(n<=12),表示给定的字符串的 ...

  8. BZOJ1195: [HNOI2006]最短母串(Trie图,搜索)

    Description 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T的子串. Input 第一行是一个正整数n(n<=12) ...

  9. bzoj1195 [HNOI2006]最短母串 AC 自动机+状压+bfs

    题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=1195 题解 建立 AC 自动机,然后构建出 trie 图. 然后直接在 trie 图上走.但是 ...

随机推荐

  1. Linux更改文件权限(二)

    更改文件权限(二)============================== (参考于千锋教育教学笔记) 命令umask [root@aminglinux ~]# umask 0022 [root@ ...

  2. Python_装饰器、迭代器、生成器

    一.装饰器 装饰器的存在是为了实现开放封闭原则: 封闭: 已实现的功能代码块不应该被修改: 开放: 对现有功能的扩展开放. 理解装饰器的三要素: 函数的作用域 高阶函数 闭包 1. 闭包 闭包定义:如 ...

  3. Java 技术栈

    JAVA是一个面向对象的编程语言,由SUN公司的程序员所开发.它不仅吸收了C++的各种优点,而且还撇弃了C++中难以理解的概念,如多继承.指针等:因此JAVA语言具有功能强大且简单易用两个特征, JA ...

  4. 【CSS】简略说明css的权重之分

    /*权重 :id > class > 标签 (小环境) 权重:内联 > 内部 > 外部 (大环境) 小环境处于内部环境中 */ <style> #p1{ /* id ...

  5. JZOJ 5809. 【NOIP2008模拟】数羊

    5809. [NOIP2008模拟]数羊 (File IO): input:sheep.in output:sheep.out Time Limits: 1000 ms  Memory Limits: ...

  6. Linux文件属性之文件权限介绍

    1)用ls -li 查看文件列表字段 红色代表的是inode 黄色代表的是文件权限 黄色里面的第一个 - 表示文件的类型(普通类型文件) d 表示目录(directory) l 表示链接文件(link ...

  7. OpenStack之Glance源码简析

    Glance简介 OpenStack镜像服务器是一套虚拟机镜像发现.注册.检索. glance架构图: Glance源码结构: glance/api:主要负责接收响应镜像管理命令的Restful请求, ...

  8. 【Unique Paths】cpp

    题目: A robot is located at the top-left corner of a m x n grid (marked 'Start' in the diagram below). ...

  9. java setVisible顺序不同导致窗体内容不显示问题

    今天学习JAVA编写窗体的时候,先写了setVisible(true);然后才去创建的各种控件以及设置大小.位置等 结果运行后只显示空白的窗体,必须最小化再最大化或点击一下边框,才显示窗体内容(即必须 ...

  10. maven文件报错(pom.xml或者jar包缺失)解决方法

    相信很多朋友在myeclipse上把maven配置好了,但是新建maven项目的时候会报错,下面我来总结以下我遇到的问题. 新建完maven项目后,pom.xml报错 1.报错的原因:很多时候我们在下 ...