数组开小毁一生……

题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=2085

这题在洛谷上有个条件是“互不包含”,其实bzoj的数据也满足这个条件,否则我目前已知的所有做法都是错的。

个人觉得AC自动机可以用其他办法做,但是没试过

KMP(或者hash), \(f[i][j]\)表示\(i\)完了接上\(j\)需要多少个字符,直接用KMP求出最长的同时是\(j\)串的前缀和\(i\)串的后缀的串即可

然后求走\((m-1)\)步的最短路,倍增Floyd

并不是多串匹配一定要用AC自动机!即使不考虑正确性,AC自动机的Trie树在复杂度上可能还多个字符集大小,不一定比KMP快,一定要合理选择!

代码

KMP:

#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#define llong long long
using namespace std; const int N = 200;
const int S = 26;
const int L = 1e5+1e3;
const int LGM = 30;
const llong INF = 1e17;
llong f[N+3][N+3],g[N+3][N+3],g0[N+3][N+3];
char a[L+3];
int nxt[L+3];
int len[L+3];
int slen[L+3];
int n; llong m; void update(llong &x,llong y) {x = min(x,y);} int KMP(char str1[],char str2[],int len1,int len2)
{
for(int i=0; i<=len1; i++) nxt[i] = 0;
for(int i=2; i<=len1; i++)
{
nxt[i] = nxt[i-1];
while(nxt[i]!=0 && str1[i]!=str1[nxt[i]+1])
{
nxt[i] = nxt[nxt[i]];
}
if(str1[i]==str1[nxt[i]+1]) nxt[i]++;
}
if(str1==str2)
{
return len1-nxt[len1];
}
else
{
int j = 0;
for(int i=1; i<=len2; i++)
{
while(j && str1[j+1]!=str2[i])
{
j = nxt[j];
}
if(j<len1 && str1[j+1]==str2[i]) j++;
}
return len1-j;
}
} int main()
{
scanf("%d%lld",&n,&m);
for(int i=1; i<=n; i++)
{
scanf("%s",a+slen[i-1]+1); len[i] = strlen(a+slen[i-1]+1); slen[i] = slen[i-1]+len[i]+1; a[slen[i]] = ' ';
}
for(int i=1; i<=n; i++) {for(int j=1; j<=n; j++) {f[i][j] = g0[i][j] = g[i][j] = INF;}}
for(int i=1; i<=n; i++) f[i][i] = len[i];
for(int i=1; i<=n; i++)
{
for(int j=1; j<=n; j++)
{
g0[j][i] = KMP(a+slen[i-1],a+slen[j-1],len[i],len[j]);
}
}
/*
for(int i=1; i<=n; i++)
{
for(int j=1; j<=n; j++)
{
printf("g0: %d %d %lld\n",i,j,g0[i][j]);
}
}
*/
m--;
while(m>0)
{
if(m&1)
{
for(int k=1; k<=n; k++)
{
for(int i=1; i<=n; i++)
{
for(int j=1; j<=n; j++)
{
update(g[i][j],f[i][k]+g0[k][j]);
}
}
}
for(int i=1; i<=n; i++)
{
for(int j=1; j<=n; j++)
{
f[i][j] = g[i][j];
g[i][j] = INF;
}
}
}
for(int k=1; k<=n; k++)
{
for(int i=1; i<=n; i++)
{
for(int j=1; j<=n; j++)
{
update(g[i][j],g0[i][k]+g0[k][j]);
}
}
}
for(int i=1; i<=n; i++)
{
for(int j=1; j<=n; j++)
{
g0[i][j] = g[i][j];
g[i][j] = INF;
}
}
m>>=1;
}
llong ans = INF;
for(int i=1; i<=n; i++)
{
for(int j=1; j<=n; j++)
{
ans = min(ans,f[i][j]);
}
}
printf("%lld\n",ans);
return 0;
}

AC自动机WA+TLE代码

#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<vector>
#define llong long long
using namespace std; const int N = 200;
const int S = 26;
const int L = 1e5+1;
const int LGM = 30;
const llong INF = 1e17;
void update(llong &x,llong y) {x = min(x,y);}
llong g0[N+3][N+3],g[N+3][N+3],f[N+3][N+3];
int len[N+3];
int fail[L+3];
int son[L+3][S+3];
int idpos[N+3];
char a[L+3];
int id[L+3];
int que[L+3];
int dep[L+3];
int n,rtn,siz; llong m; void insertstr(char str[],int sid)
{
int u = rtn;
for(int i=1; i<=len[sid]; i++)
{
if(son[u][str[i]])
{
u = son[u][str[i]];
}
else
{
siz++; son[u][str[i]] = siz;
u = siz;
}
}
id[u] = sid;
idpos[sid] = u;
} void add(int u,int v,llong w)
{
update(g0[id[u]][id[v]],w);
} void getfail()
{
int head = 1,tail = 0;
for(int i=1; i<=S; i++)
{
int v = son[rtn][i];
if(v)
{
tail++; que[tail] = v;
fail[v] = rtn;
}
}
while(head<=tail)
{
int u = que[head]; head++;
for(int i=1; i<=S; i++)
{
int v = son[u][i];
if(v) {fail[son[u][i]] = son[fail[u]][i]; tail++; que[tail] = v;}
else {son[u][i] = son[fail[u]][i];}
}
}
} void bfs(int s)
{
printf("bfs(%d)\n",s);
for(int i=1; i<=siz; i++) dep[i] = 0;
int head = 1,tail = 1; que[1] = s;
while(head<=tail)
{
int u = que[head]; head++;
int v = fail[i];
for(int i=1; i<=S; i++)
{
v = son[u][i];
if(v && dep[v]==0)
{
dep[v] = dep[u]+1;
add(s,v,dep[v]);
tail++; que[tail] = v;
}
}
}
} int main()
{
scanf("%d%lld",&n,&m); rtn = siz = 0;
for(int i=1; i<=n; i++)
{
scanf("%s",a+1); len[i] = strlen(a+1);
for(int j=1; j<=len[i]; j++) a[j] -= 96;
insertstr(a,i);
}
getfail();
for(int i=0; i<=siz; i++) printf("AC%d fail%d\n",i,fail[i]);
for(int i=0; i<=siz; i++) {for(int j=1; j<=S; j++) {if(son[i][j]) {printf("son[%d][%c]=%d\n",i,j+96,son[i][j]);}}}
for(int i=1; i<=n; i++) {for(int j=1; j<=n; j++) {f[i][j] = g0[i][j] = g[i][j] = INF;}}
for(int i=1; i<=n; i++) f[i][i] = 0ll;
for(int i=1; i<=siz; i++)
{
if(id[i])
{
bfs(i);
}
}
/*
for(int i=1; i<=n; i++)
{
for(int j=1; j<=n; j++)
{
printf("g0: %d %d %lld\n",i,j,g0[i][j]);
}
}
*/
m--;
while(m>0)
{
if(m&1)
{
for(int k=1; k<=n; k++)
{
for(int i=1; i<=n; i++)
{
for(int j=1; j<=n; j++)
{
update(g[i][j],f[i][k]+g0[k][j]);
}
}
}
for(int i=1; i<=n; i++)
{
for(int j=1; j<=n; j++)
{
f[i][j] = g[i][j];
g[i][j] = INF;
}
}
}
for(int k=1; k<=n; k++)
{
for(int i=1; i<=n; i++)
{
for(int j=1; j<=n; j++)
{
update(g[i][j],g0[i][k]+g0[k][j]);
}
}
}
for(int i=1; i<=n; i++)
{
for(int j=1; j<=n; j++)
{
g0[i][j] = g[i][j];
g[i][j] = INF;
}
}
m>>=1;
}
llong ans = INF;
for(int i=1; i<=n; i++)
{
for(int j=1; j<=n; j++)
{
ans = min(ans,len[i]+f[i][j]);
}
}
printf("%lld\n",ans);
return 0;
}
/*
4 11
aaaaaaa
aaaa
aaaaaa
aaaaaaaaaa
*/

BZOJ 2085 luogu P3502 [POI2010]Hamsters (KMP、Floyd、倍增)的更多相关文章

  1. 洛谷P3502 [POI2010]CHO-Hamsters感想及题解(图论+字符串+矩阵加速$dp\&Floyd$)

    洛谷P3502 [POI2010]CHO-Hamsters感想及题解(图论+字符串+矩阵加速\(dp\&Floyd\)) 标签:题解 阅读体验:https://zybuluo.com/Junl ...

  2. [BZOJ 1535] [Luogu 3426]SZA-Template (KMP+fail树+双向链表)

    [BZOJ 1535] [Luogu 3426]SZA-Template (KMP+fail树+双向链表) 题面 Byteasar 想在墙上涂一段很长的字符,他为了做这件事从字符的前面一段中截取了一段 ...

  3. BZOJ 3052/Luogu P4074 [wc2013]糖果公园 (树上带修莫队)

    题面 中文题面,难得解释了 BZOJ传送门 Luogu传送门 分析 树上带修莫队板子题... 开始没给分块大小赋初值T了好一会... CODE #include <bits/stdc++.h&g ...

  4. BZOJ 3931 / Luogu P3171 [CQOI2015]网络吞吐量 (最大流板题)

    题面 中文题目,不解释: BZOJ传送门 Luogu传送门 分析 这题建图是显然的,拆点后iii和i′i'i′连容量为吞吐量的边,根据题目要求,111和nnn的吞吐量看作∞\infty∞. 然后用di ...

  5. BZOJ 3894 / Luogu P4313 文理分科 (拆点最小割)

    题面 中文题面- BZOJ 传送门 Luogu 传送门 分析 这道题类似于BZOJ 3774 最优选择,然后这里有一篇博客写的很好- Today_Blue_Rainbow's Blog 应该看懂了吧- ...

  6. BZOJ 2039 / Luogu P1791 [2009国家集训队]employ人员雇佣 (最小割)

    题面 BZOJ传送门 Luogu传送门 分析 考虑如何最小割建图,因为这仍然是二元关系,我们可以通过解方程来确定怎么建图,具体参考论文 <<浅析一类最小割问题 湖南师大附中 彭天翼> ...

  7. BZOJ 2127 / Luogu P1646 [国家集训队]happiness (最小割)

    题面 BZOJ传送门 Luogu传送门 分析 这道题又出现了二元关系,于是我们只需要解方程确定怎么连边就行了 假设跟SSS分在一块是选文科,跟TTT分在一块是选理科,先加上所有的收益,再来考虑如何让需 ...

  8. [BZOJ 3295] [luogu 3157] [CQOI2011]动态逆序对(树状数组套权值线段树)

    [BZOJ 3295] [luogu 3157] [CQOI2011] 动态逆序对 (树状数组套权值线段树) 题面 给出一个长度为n的排列,每次操作删除一个数,求每次操作前排列逆序对的个数 分析 每次 ...

  9. [BZOJ 3110] [luogu 3332] [ZJOI 2013]k大数查询(权值线段树套线段树)

    [BZOJ 3110] [luogu 3332] [ZJOI 2013]k大数查询(权值线段树套线段树) 题面 原题面有点歧义,不过从样例可以看出来真正的意思 有n个位置,每个位置可以看做一个集合. ...

随机推荐

  1. 【译文】利用STAN做贝叶斯回归分析:Part 2 非正态回归

    [译文]利用STAN做贝叶斯回归分析:Part 2 非正态回归 作者 Lionel Hertzogn 前一篇文章已经介绍了怎样在R中调用STAN对正态数据进行贝叶斯回归.本文则将利用三个样例来演示怎样 ...

  2. 弹出框中选项卡的运用(easyUI)

    先看一下页面效果: 此处有两个知识点:一个是弹出框的运用,一个是选项卡的运用 分析一下该HTML代码,最外面一个div是弹出框的,默认是关闭状态,可通过ID来控制弹出框的开关,该div的样式是easy ...

  3. B1202 [HNOI2005]狡猾的商人 并查集

    其实就是并查集的题.维护一个前缀和,然后用并查集维护前缀和,每次判断是否合理就行了. 题干: Description 刁姹接到一个任务,为税务部门调查一位商人的账本,看看账本是不是伪造的.账本上记录了 ...

  4. Linux防火墙设置

    对于Internet上的系统,不管是什么情况都要明确一点:网络是不安全的.因此,虽然创建一个防火墙并不能保证系统100%安全,但却是绝对必要的. Linux提供了一个非常优秀的防火墙工具-netfil ...

  5. ubuntu Ngin Install

    安装gcc g++的依赖库 #apt-get install build-essential #apt-get install libtool 安装 pcre依赖库 #sudo apt-get upd ...

  6. WPF TextBox 仅允许输入数字

    因为在 IValueConverter 实现中,当文本不能转换为目标类型时返回 DependencyProperty.UnsetValue ,Validation.GetHasError 返回 tru ...

  7. JavaScript异步加载方案

    (1) defer,只支持IE defer属性的定义和用法(我摘自w3school网站) defer 属性规定是否对脚本执行进行延迟,直到页面加载为止. 有的 javascript 脚本 docume ...

  8. HDFS Shell命令操作与java代码操作

    (一)编程实现以下功能,并利用 Hadoop 提供的 Shell 命令完成相同任务: (1)     向 HDFS 中上传任意文本文件,如果指定的文件在 HDFS 中已经存在,则由用户来指定是追加到原 ...

  9. JavaScript中比较运算符的使用

    比较运算符的基本操作过程是:首先对操作数进行比较,这个操作数可以是数字也可以是字符串,然后返回一个布尔值true或false. 在JavaScript中常用的比较运算符如下表所示. 例如,某商场店庆搞 ...

  10. DB2数据常用指令

    ************************************************************** 这个只是个人平时总结,如果有更好的欢迎告诉我,一起学习一起成长 ***** ...