正题

题目链接:https://www.luogu.com.cn/problem/P4022


题目大意

给出\(m\)个模板串。

然后\(n\)次询问给出一个串\(S\)要求找到一个最大的\(L\)使得能够将\(S\)超过\(90\%\)的部分拿出来分后每个串都是某个模板串的子串且长度不小于\(L\)。

所有输入文件长度不超过 \(1100000\) 字节。字符集为\(\{0,1\}\)


解题思路

先把模板串拿出来构一个广义SAM,然后考虑用这个对串进行匹配。

先对于每个位置求出一个\(len_i\)表示一个最长的长度使得\(i\)的后缀是某个模板串的子串。

然后考虑二分一个\(L\)后进行\(dp\)。

那么有

\[f_i=max\{f_{i-1},f_j+i-j\}(\ j\in[i-len_i,i-L)\ )
\]

因为\(i-len_i\)单调所以把\(j\)丢进单调队列里就好了。

时间复杂度\(O(n\log n)\)


code

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int N=1200000;
int n,m,cnt,f[N],q[N],ml[N];
int ch[N][2],len[N],fa[N];
char s[N];
int Insert(int p,int c){
if(ch[p][c]){
int q=ch[p][c];
if(len[p]+1==len[q])return q;
else{
int nq=++cnt;len[nq]=len[p]+1;
memcpy(ch[nq],ch[q],sizeof(ch[nq]));
fa[nq]=fa[q];fa[q]=nq;
for(;p&&ch[p][c]==q;p=fa[p])ch[p][c]=nq;
return nq;
}
}
int np=++cnt;len[np]=len[p]+1;
for(;p&&!ch[p][c];p=fa[p])ch[p][c]=np;
if(!p)fa[np]=1;
else{
int q=ch[p][c];
if(len[p]+1==len[q])fa[np]=q;
else{
int nq=++cnt;len[nq]=len[p]+1;
memcpy(ch[nq],ch[q],sizeof(ch[nq]));
fa[nq]=fa[q];fa[q]=fa[np]=nq;
for(;p&&ch[p][c]==q;p=fa[p])ch[p][c]=nq;
}
}
return np;
}
bool check(int L,int n){
int head=1,tail=0,ans=0;
for(int i=1;i<=n;i++){
if(i>=L){
int j=i-L;
while(head<=tail&&f[j]-j>f[q[tail]]-q[tail])tail--;
q[++tail]=j;
}
while(head<=tail&&q[head]<i-ml[i])head++;
f[i]=0;
if(head<=tail)f[i]=f[q[head]]+i-q[head];
f[i]=max(f[i],f[i-1]);
ans=max(ans,f[i]);
}
return (ans*10>=n*9);
}
int main()
{
scanf("%d%d",&n,&m);cnt=1;
for(int i=1;i<=m;i++){
scanf("%s",s+1);
int l=strlen(s+1),x=1;
for(int j=1;j<=l;j++)
x=Insert(x,s[j]-'0');
}
while(n--){
scanf("%s",s+1);
int sl=strlen(s+1),x=1,L=0;
for(int i=1;i<=sl;i++){
int c=s[i]-'0';
while(x&&!ch[x][c])
{x=fa[x];L=len[x];}
if(x)x=ch[x][c],L++;
else x=1,L=0;
ml[i]=L;
}
int l=1,r=sl;
while(l<=r){
int mid=(l+r)>>1;
if(check(mid,sl))l=mid+1;
else r=mid-1;
}
printf("%d\n",r);
}
return 0;
}

最新文章

  1. 最近发现了个js传图预览的函数和大家分享下
  2. DockerProblem
  3. js日历插件 中文、英文日历
  4. Mac nginx PCRE install ngnix
  5. CentOS7 network
  6. iOS-画板的实现
  7. 从文章&quot;避免复制与粘贴&quot;到文章&quot;Extract Method&quot;的反思(3)
  8. Google改变生活
  9. BigDecimal用法详解(转)
  10. LIQN join on 后跟多个条件
  11. php写留言板
  12. 【FFmpeg】FFmpeg常用基本命令
  13. Python执行系统命令:使用subprocess的Popen函数
  14. 绑定微信以及获取openId
  15. MySQL5.7在JSON解析后丢失小数部分的Bug
  16. c# 之 事务
  17. MSC服务器-主从检测脚本-check_server_state.sh
  18. AWS机器学习初探(1):Comprehend - 自然语言处理服务
  19. DMZ原理与应用
  20. 64位Windows操作系统中的注冊表

热门文章

  1. SpringBoot快速入门(一)
  2. Docker创建Docker-Registry-私服
  3. Centos7上yum安装mongodb4-2
  4. windows上解决git每次重复输入账号密码
  5. ant的javac任务的相关属性配置
  6. Ubuntu 16.04 + python3 源码 安装+使用labelImg最新版
  7. MySQL 事务和锁
  8. 用GUI实现java版贪吃蛇小游戏
  9. 发布日志 - kratos v2.0.4 版本发布
  10. mpvue 生成字节跳动小程序的问题!!