BZOJ1559 [JSOI2009]密码 【AC自动机 + 状压dp】
题目链接
题解
考虑到这是一个包含子串的问题,而且子串非常少,我们考虑\(AC\)自动机上的状压\(dp\)
设\(f[i][j][s]\)表示长度为\(i\)的串,匹配到了\(AC\)自动机\(j\)号节点,且已匹配集合为\(s\)的方案数
直接在\(AC\)自动机上转移即可
但是为了防止使用\(last\)指针之类的,计算匹配的串,我们先将原串的集合去重和去包含关系
方案怎么办?
考虑到\(ans \le 42\),一定是刚好若干个原串以最长前后缀相同的方式相接
因为如果不是以最长前后缀的方式相接,那么一定存在更小的方案,这个时候就会多出若干空位置可以随便填
一个位置可以填\(26\)种字母,而且可以放在前缀或者后缀,所以有一个空位置就有\(52\)种方案,大于\(42\)种
所以可以放心\(O(n!)\)枚举排列相接
\(1A\)了很开心
#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
#include<string>
#include<cmath>
#include<queue>
#include<map>
#define Redge(u) for (int k = h[u],to; k; k = ed[k].nxt)
#define REP(i,n) for (int i = 1; i <= (n); i++)
#define mp(a,b) make_pair<int,int>(a,b)
#define cls(s) memset(s,0,sizeof(s))
#define cp pair<int,int>
#define LL long long int
using namespace std;
const int maxn = 105,maxm = 100005,INF = 1000000000;
inline int read(){
int out = 0,flag = 1; char c = getchar();
while (c < 48 || c > 57){if (c == '-') flag = -1; c = getchar();}
while (c >= 48 && c <= 57){out = (out << 3) + (out << 1) + c - 48; c = getchar();}
return out * flag;
}
int ch[maxn][26],fail[maxn],tag[maxn],cnt,len[12];
char s[12][12];
int isin[maxn];
int same[12][12];
LL f[26][105][1 << 10];
void ins(int p){
int u = 0,id;
for (int i = 1; i <= len[p]; i++){
id = s[p][i] - 'a';
u = ch[u][id] ? ch[u][id] : (ch[u][id] = ++cnt);
}
tag[u] |= (1 << p - 1);
}
void getf(){
queue<int> q;
for (int i = 0; i < 26; i++) if (ch[0][i]) q.push(ch[0][i]);
int u,v;
while (!q.empty()){
u = q.front(); q.pop();
for (int i = 0; i < 26; i++){
v = ch[u][i];
if (!v){
ch[u][i] = ch[fail[u]][i];
continue;
}
fail[v] = ch[fail[u]][i];
q.push(v);
}
}
}
int L,n;
int check(int u,int v){
if (len[u] > len[v]) return 0;
if (len[u] == len[v]){
for (int i = 1; i <= len[u]; i++)
if (s[u][i] != s[v][i])
return 0;
return 1;
}
for (int i = 1; i <= len[v] - len[u] + 1; i++){
int flag = true;
for (int j = 1; j <= len[u]; j++){
if (s[u][j] != s[v][i + j - 1]){
flag = false;
break;
}
}
if (flag) return 2;
}
return 0;
}
int cal(int u,int v){
for (int i = max(1,len[u] - len[v] + 1); i <= len[u]; i++){
int flag = true;
for (int j = 1; i + j - 1 <= len[u]; j++)
if (s[u][i + j - 1] != s[v][j]){
flag = false;
break;
}
if (flag) return len[u] - i + 1;
}
return 0;
}
int vis[maxn],a[maxn],ansi,Ltot;
struct node{
char s[30];
int len;
}ans[50];
inline bool operator <(const node& a,const node& b){
return strcmp(a.s + 1,b.s + 1) < 0;
}
void Check(){
int sum = Ltot;
for (int i = 1; i < n; i++)
sum -= same[a[i]][a[i + 1]];
if (sum == L){
ansi++;
for (int i = 1; i <= n; i++){
int u = a[i];
for (int j = same[a[i - 1]][u] + 1; j <= len[u]; j++){
ans[ansi].s[++ans[ansi].len] = s[u][j];
}
}
}
}
void dfs(int u){
if (u > n){
Check();
return;
}
for (int i = 1; i <= n; i++)
if (!vis[i]){
vis[i] = true;
a[u] = i;
dfs(u + 1);
vis[i] = false;
}
}
void work(){
for (int i = 1; i <= n; i++)
for (int j = 1; j <= n; j++)
if (i != j) same[i][j] = cal(i,j);
for (int i = 1; i <= n; i++) Ltot += len[i];
dfs(1);
sort(ans + 1,ans + 1 + ansi);
for (int i = 1; i <= ansi; i++)
printf("%s\n",ans[i].s + 1);
}
int main(){
L = read(); n = read();
for (int i = 1; i <= n; i++){
scanf("%s",s[i] + 1);
len[i] = strlen(s[i] + 1);
}
for (int i = 1; i <= n; i++)
for (int j = 1; j <= n; j++)
if (i != j ){
int t = check(i,j);
if (t == 2 || (t == 1 && i > j))
isin[i] = true;
}
int tot = 0;
for (int i = 1; i <= n; i++)
if (!isin[i]){
strcpy(s[++tot] + 1,s[i] + 1);
len[tot] = len[i];
ins(tot);
}
n = tot;
getf();
int maxv = (1 << n) - 1,u;
f[0][0][0] = 1;
for (int i = 0; i < L; i++){
for (int j = 0; j <= cnt; j++){
for (int s = 0; s <= maxv; s++){
if (!f[i][j][s]) continue;
for (int k = 0; k < 26; k++){
u = ch[j][k];
f[i + 1][u][s | tag[u]] += f[i][j][s];
}
}
}
}
LL ans = 0;
for (int i = 0; i <= cnt; i++)
ans += f[L][i][maxv];
printf("%lld\n",ans);
if (ans <= 42) work();
return 0;
}
BZOJ1559 [JSOI2009]密码 【AC自动机 + 状压dp】的更多相关文章
- BZOJ 1559: [JSOI2009]密码( AC自动机 + 状压dp )
建AC自动机后, dp(x, y, s)表示当前长度为x, 在结点y, 包括的串的状态为s的方案数, 转移就在自动机上走就行了. 对于输出方案, 必定是由给出的串组成(因为<=42), 所以直接 ...
- hdu 2825 aC自动机+状压dp
Wireless Password Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others ...
- HDU 3247 Resource Archiver(AC自动机 + 状压DP + bfs预处理)题解
题意:目标串n( <= 10)个,病毒串m( < 1000)个,问包含所有目标串无病毒串的最小长度 思路:貌似是个简单的状压DP + AC自动机,但是发现dp[1 << n][ ...
- zoj3545Rescue the Rabbit (AC自动机+状压dp+滚动数组)
Time Limit: 10 Seconds Memory Limit: 65536 KB Dr. X is a biologist, who likes rabbits very much ...
- hdu2825 Wireless Password(AC自动机+状压dp)
Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Submission ...
- hdu 4057--Rescue the Rabbit(AC自动机+状压DP)
题目链接 Problem Description Dr. X is a biologist, who likes rabbits very much and can do everything for ...
- [BZOJ1559]密码 AC自动机+状压
问题 K: [JSOI2009]密码 时间限制: 1 Sec 内存限制: 64 MB 题目描述 众所周知,密码在信息领域起到了不可估量的作用.对于普通的登陆口令,唯一的破解 方法就是暴力破解一逐个尝 ...
- BZOJ1559[JSOI2009]密码——AC自动机+DP+搜索
题目描述 输入 输出 样例输入 10 2 hello world 样例输出 2 helloworld worldhello 提示 这题算是一个套路题了,多个串求都包含它们的长为L的串的方案数. 显然是 ...
- hdu 6086 -- Rikka with String(AC自动机 + 状压DP)
题目链接 Problem Description As we know, Rikka is poor at math. Yuta is worrying about this situation, s ...
随机推荐
- Docker与FastDFS的安装命令及使用
Docker特点 1)上手快 用户只需要几分钟,就可以把自己的程序“Docker 化”.Docker 依赖于“写时复制” (copy-on-write)模型,使修改应用程序也非常迅速,可以说达到“随心 ...
- 帆软中使用switch将控件的显示值“传递”给单元格
如下图,控件的实际值和显示值是我们自定义的. 当我们选择控件时,想要在某个单元格内显示控件的显示值.一般我们在单元格内直接 $控件名 可以获得控件值.比如当我们选择事故数时,我们自然不能在单元格内直 ...
- ruby 数据类型Range
范围(Range)无处不在:a 到 z. 0 到 9.等等.Ruby 支持范围,并允许我们以不同的方式使用范围: 作为序列的范围 作为条件的范围 作为间隔的范围 作为序列的范围 (1..5) #==& ...
- C中 snprintf()函数的作用
函数原型:int snprintf(char* dest_str,size_t size,const char* format,...); 函数功能:先将可变参数 “…” 按照format的格式格式化 ...
- 【Python让生活更美好01】os与shutil模块的常用方法总结
Python作为一种解释型的高级语言,脚本语言,又被称作“胶水语言”,就是因为其灵活的语法和其依靠浩如烟海的第三方包实现的丰富多彩的功能,而os和shutil就是这样一种功能强大的模块,可以非常快捷地 ...
- Andy's First Dictionary(uva 10815) set用法
参考:https://www.cnblogs.com/yjlblog/p/6947747.html https://blog.csdn.net/hnust_taoshiqian/article/det ...
- 最小费用最大流模板 洛谷P3381
题目描述 如题,给出一个网络图,以及其源点和汇点,每条边已知其最大流量和单位流量费用,求出其网络最大流和在最大流情况下的最小费用. 输入输出格式 输入格式: 第一行包含四个正整数N.M.S.T,分别表 ...
- python2.7练习小例子(九)
9)1.题目:暂停一秒输出. 程序分析:使用 time 模块的 sleep() 函数. 程序源代码: #!/usr/bin/python # -*- coding: UTF-8 ...
- 使用USB Key(加密狗)实现身份认证
首先你需要去买一个加密狗设备,加密狗是外形酷似U盘的一种硬件设备! 这里我使用的坚石诚信公司的ET99产品 公司项目需要实现一个功能,就是客户使用加密狗登录, 客户不想输入任何密码之类的东西,只需要插 ...
- Linux的系统安全设置Shell脚本
#!/bin/sh # desc: setup linux system security # powered by www.lvtao.net #account setup passwd -l xf ...