bzoj 1195: [HNOI2006]最短母串【状压dp】
我有病吧……明明直接枚举是否匹配就可以非要写hash,然后果然冲突了(……我个非酋居然还敢用hash
设f[s][i]为已选串状态为s并且最后一个串是i,还有预处理出g[i][j]表示最长有长为g[i][j]的i串后缀等于j串前缀这里,直接暴力匹配即可……
然后注意到比较麻烦的事要求字典序最小,但是因为空间限制我们又不能给每个f存一个串,所以我们设t[s][i][j]为状态使f[s][i]长度最小且字典序最小的选串顺序的第j个选的是那个串,这个更新f[s][i]的时候直接更新,注意如果有等于当前f[s][i]的情况,要分别把旧的t[s][i]和当前情况对应的两个串都求出来比一下字典序
长度dp部分随便转移一下就好
#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
const int N=15;
int n,f[5005][N],t[5005][N][N],g[N][N],len[N],ans=1e9,b[N];
bool v[N];
char c[N][55],s1[1005],s2[1005],a[1005];
struct qwe
{
int l;
char c[55];
};
bool cmp(const qwe &a,const qwe &b)
{
for(int i=1;i<=min(a.l,b.l);i++)
if(a.c[i]!=b.c[i])
return a.c[i]<b.c[i];
return a.l<b.l;
}
bool ok(int i,int j,int l)
{
for(int k=1;k<=l;k++)
if(c[i][len[i]-l+k]!=c[j][k])
return 0;
return 1;
}
int main()
{
scanf("%d",&n);
for(int i=1;i<=n;i++)
{
scanf("%s",c[i]+1);
len[i]=strlen(c[i]+1);
b[i]=1<<(i-1);
}
for(int i=1;i<=n;i++)
for(int j=1;j<=n;j++)
for(int l=min(len[i],len[j]);l>=0;l--)
if(ok(i,j,l))
{
g[i][j]=l;
break;
}
// dfs(1,0);
memset(f,0x3f,sizeof(f));
int inf=f[0][0];
for(int i=1;i<=n;i++)
f[b[i]][i]=len[i],t[b[i]][i][1]=i;
for(int s=1,le=(1<<n)-1;s<=le;s++)
{
int con=0;
for(int i=1;i<=n;i++)
if(s&b[i])
con++;
for(int i=1;i<=n;i++)
if(s&b[i])
{
for(int j=1;j<=n;j++)
if(!(s&b[j]))
{
int nw=f[s][i]+len[j]-g[i][j],w=(s|b[j]);
if(nw<f[w][j])
{
f[w][j]=nw;
for(int k=1;k<=con;k++)
t[w][j][k]=t[s][i][k];
t[w][j][con+1]=j;
}
else if(nw==f[w][j])
{
int t1=0,t2=0,fl=0;
for(int k=1;k<=con+1;k++)
for(int l=1+g[t[w][j][k-1]][t[w][j][k]];l<=len[t[w][j][k]];l++)
s1[++t1]=c[t[w][j][k]][l];
for(int k=1;k<=con;k++)
for(int l=1+g[t[s][i][k-1]][t[s][i][k]];l<=len[t[s][i][k]];l++)
s2[++t2]=c[t[s][i][k]][l];
for(int l=1+g[t[s][i][con]][j];l<=len[j];l++)
s2[++t2]=c[j][l];
for(int k=1;k<=nw;k++)
{
if(s1[k]<s2[k])
break;
if(s1[k]>s2[k])
{
fl=1;
break;
}
}
if(fl)
{
for(int k=1;k<=con;k++)
t[w][j][k]=t[s][i][k];
t[w][j][con+1]=j;
}
}
}
}
}
// printf("%d\n",ans);
int t1=0,w=(1<<n)-1,ans=1e9;
for(int j=1;j<=n;j++)
{
if(f[w][j]<ans)
{
ans=f[w][j];
int t1=0;
for(int k=1;k<=n;k++)
for(int l=1+g[t[w][j][k-1]][t[w][j][k]];l<=len[t[w][j][k]];l++)
a[++t1]=c[t[w][j][k]][l];
}
else if(f[w][j]==ans)
{
int t1=0,fl=0;
for(int k=1;k<=n;k++)
for(int l=1+g[t[w][j][k-1]][t[w][j][k]];l<=len[t[w][j][k]];l++)
s1[++t1]=c[t[w][j][k]][l];
for(int k=1;k<=ans;k++)
{
if(a[k]<s1[k])
break;
if(a[k]>s1[k])
{
fl=1;
break;
}
}
if(fl)
for(int k=1;k<=ans;k++)
a[k]=s1[k];
}
}
for(int i=1;i<=ans;i++)
printf("%c",a[i]);
return 0;
}
bzoj 1195: [HNOI2006]最短母串【状压dp】的更多相关文章
- bzoj 1195: [HNOI2006]最短母串 爆搜
1195: [HNOI2006]最短母串 Time Limit: 10 Sec Memory Limit: 32 MBSubmit: 894 Solved: 288[Submit][Status] ...
- BZOJ 1195: [HNOI2006]最短母串
1195: [HNOI2006]最短母串 Time Limit: 10 Sec Memory Limit: 32 MBSubmit: 1346 Solved: 450[Submit][Status ...
- bzoj 1195 [HNOI2006]最短母串 bfs 状压 最短路 AC自动机
LINK:最短母串 求母串的问题.不适合SAM. 可以先简化问题 考虑给出的n个字符串不存在包含关系. 那么 那么存在的情况 只可能有 两个字符串拼接起来能表示另外一个字符串 或者某个字符串的后缀可以 ...
- BZOJ 1195 [HNOI2006]最短母串 (Trie图+状压+bfs最短路)
BZOJ1195 LOJ10061 题目大意:给你$n$个模式串,求一个最短且字典序最小的文本串并输出这个串,$n<=12,len<=50$ 首先对所有模式串构造$Trie$图,$Trie ...
- BZOJ 1195: [HNOI2006]最短母串 AC自动机+状压+搜索
思路比较直接. 由于 $n$ 很小,直接定义 $f[i][j]$ 表示当前在自动机中的节点 $i,$ 被覆盖串的集合为 $j$ 的方案数. #include <bits/stdc++.h> ...
- 【刷题】BZOJ 1195 [HNOI2006]最短母串
Description 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T的子串. Input 第一行是一个正整数n(n<=12) ...
- 【状态压缩dp】1195: [HNOI2006]最短母串
一个清晰的思路就是状压dp:不过也有AC自动机+BFS的做法 Description 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T ...
- 1195: [HNOI2006]最短母串 - BZOJ
Description 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T的子串.Input 第一行是一个正整数n(n<=12), ...
- 1195: [HNOI2006]最短母串
思路:好像以前谁问过我这题... 状个压就好啦, 把包含在其他串中的字符串删掉, 预处理除每两个字符串之间的关系, dp[ state ][ i ] 表示在state的状态下, 最后一个字符串是第i ...
随机推荐
- POJ 1144 割点
题意 :求割点的数量 #include<iostream> #include<stdio.h> #include<vector> #include<strin ...
- poj -1185 炮兵阵地 (经典状压dp)
http://poj.org/problem?id=1185 参考博客:http://poj.org/problem?id=1185 大神博客已经讲的很清楚了,注意存状态的时候是从1开始的,所以初始化 ...
- [洛谷U22156]未曾届到游览(矩阵树定理)
题目背景 又到了某任*堂开关中学一年一度的自主招生考试的时间了,在考试完后许多家长决定带着自己的孩子参观一下这所距千年名校还有890周年的百年学校: 题目描述 这所学校的布局非常奇怪,是一个由N 个点 ...
- 战五渣系列之八(绝杀AOP)
开发不用aop.程序猿的人生该会浪费多少时间.我想是时候让程序猿打败alpha狗了.程序猿解救世界. 1.概念 面向切面编程.这意味着,一切不在流水线上的东西.包含权限.日志.缓存.校验.资源.事物. ...
- cocos2d-x 2.0下怎样让BOX2D DEBUG DRAW的方法笔记
原文链接: 这两天玩 cocos2d-x 和 box2d,发现 cocos2d-x 2.0 版本号要使用老方法 debug 渲染会出错.于是找到了新方法来 debug draw: 首先在你的头文件中添 ...
- Spark应用远程调试
本来想用Eclipse的.然而在网上找了一圈,发现大家都在说IntelliJ怎样怎样好.我也受到了鼓励,遂决定在这台破机器上鼓捣一次IntelliJ吧. Spark程序远程调试,就是将本地IDE连接到 ...
- mysql中“Table ‘’ is read only”的解决办法
之前是在linux下面直接Copy的data下面整个数据库文件夹,在phpMyAdmin里面重新赋予新用户相应权限后,drupal成功连接上数据库.但出现N多行错误提示,都是跟Cache相关的表是‘R ...
- oracle 正则查询json返回报文中某个字段的值
接口返回报文为json 格式,如下: {"body":{"businessinfo":{"c1rate":"25.00" ...
- web.xml中的ServletContextListener
要想了解ServletContextListener,先看看web.xml中的<listener>配置. 一)web.xml中的内容载入顺序: 首先能够肯定的是,载入顺序与它们在 web. ...
- web 界面设计---大道至简
http://www.cnblogs.com/coder2012/p/4023442.html 一个非常精简的webpy页面博客 qing.weibo.com 新浪的轻微博也不错精简