BZOJ 2806: [Ctsc2012]Cheat(单调队列优化dp+后缀自动机)
解题思路
肯定先要建出来广义后缀自动机。刚开始以为是个二分+贪心,写了一下结果\(20\)分。说一下正解,首先显然\(L_0\)具有单调性,是可以二分的。考虑二分后怎样判合法,对于分割序列很容易想到\(dp\),设\(f_i\)表示前\(i\)个字符匹配成功数量,那么有转移方程\(f_i=max(f_j+i-j)(i-j>=L\) 且 \(j\)到\(i\)可以匹配 \()\),\(L\)是二分出来的限制,判断是否能匹配可以预处理,预处理出\(mth_i\)表示\(i\)最多能与往前\(mth_i\)位匹配成功,那么第二个条件就变成了\(j>=i-mth_i\)。如果这样做是\(O(n^2logn)\)的,实测可以拿到\(75\)分2333。考虑优化,发现\(i-mth_i\)具有单调性,因为每移动一格\(i\)会\(+1\),而\(mth_i\)最多\(+1\)。那么可以用一个单调递减队列来优化,每次将\(i-lim\)入队,取出队头更新答案,时间复杂度为\(O(nlogn)\)。
代码
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
using namespace std;
const int N=1100005<<1;
int n,m,res,ans,f[N],mth[N],q[N],hd,tl;
char s[N];
struct SAM{
int ch[N][2],fa[N],l[N],lst,cnt;
void Insert(int c){
int p=lst,np=++cnt; l[np]=l[p]+1; lst=cnt;
for(;p && !ch[p][c];p=fa[p]) ch[p][c]=np;
if(!p) fa[np]=1;
else {
int q=ch[p][c];
if(l[p]+1==l[q]) fa[np]=q;
else {
int nq=++cnt; l[nq]=l[p]+1;
memcpy(ch[nq],ch[q],sizeof(ch[nq]));
fa[nq]=fa[q]; fa[q]=fa[np]=nq;
for(;ch[p][c]==q;p=fa[p]) ch[p][c]=nq;
}
}
}
void prework(int len){
int now=1,num=0;
for(int i=1;i<=len;i++){
if(ch[now][s[i]-'0']) now=ch[now][s[i]-'0'],num++;
else {
for(;now && !ch[now][s[i]-'0'];now=fa[now]);
if(!now) now=1,num=0;
else num=l[now]+1,now=ch[now][s[i]-'0'];
}
mth[i]=num;
}
}
bool check(int lim,int len){
int tmp; hd=1; tl=0;
for(int i=1;i<=len;i++){
f[i]=f[i-1]; if(i<lim) continue; tmp=i-mth[i];
while(hd<=tl && f[q[tl]]-q[tl]<=f[i-lim]-i+lim) tl--;
q[++tl]=i-lim;
while(hd<=tl && q[hd]<tmp) hd++;
if(hd<=tl) f[i]=max(f[i],f[q[hd]]-q[hd]+i);
}
// for(int i=1;i<=len;i++){
// f[i]=f[i-1];
// for(int j=max(0,i-mth[i]);j+lim<=i;j++)
// f[i]=max(f[i],f[j]+i-j);
// }
return len-f[len]>res?0:1;
}
void solve(int len){
prework(len);
int L=1,R=len,mid;
while(L<=R){
mid=(L+R)>>1;
if(check(mid,len)) L=mid+1,ans=mid;
else R=mid-1;
}
printf("%d\n",ans);
}
}sam;
int main(){
scanf("%d%d",&n,&m); sam.cnt=1; int len;
for(int i=1;i<=m;i++){
scanf("%s",s+1); sam.lst=1;
len=strlen(s+1);
for(int j=1;j<=len;j++) sam.Insert(s[j]-'0');
}
for(int i=1;i<=n;i++){
scanf("%s",s+1); len=strlen(s+1);
res=(len*9+9)/10; res=len-res;
sam.solve(len); ans=0;
}
return 0;
}
BZOJ 2806: [Ctsc2012]Cheat(单调队列优化dp+后缀自动机)的更多相关文章
- bzoj 2216: Lightning Conductor 单调队列优化dp
题目大意 已知一个长度为\(n\)的序列\(a_1,a_2,...,a_n\)对于每个\(1\leq i\leq n\),找到最小的非负整数\(p\)满足: 对于任意的\(j\), \(a_j \le ...
- BZOJ 1233 干草堆 (单调队列优化DP)
$ BZOJ~1233~~ $ 干草堆: (题目特殊性质) $ solution: $ 很妙的一道题目,开始看了一眼觉得是个傻逼贪心,从后往前当前层能多短就多短,尽量节省花费.但是这是DP专题,怎么会 ...
- BZOJ 1855 股票交易(单调队列优化DP)
题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=1855 题意:最近lxhgww又迷上了投资股票, 通过一段时间的观察和学习,他总结出了股票 ...
- BZOJ 1499 [NOI2005] 瑰丽华尔兹 | 单调队列优化DP
BZOJ 1499 瑰丽华尔兹 | 单调队列优化DP 题意 有一块\(n \times m\)的矩形地面,上面有一些障碍(用'#'表示),其余的是空地(用'.'表示).每时每刻,地面都会向某个方向倾斜 ...
- bzoj 1499 [NOI2005]瑰丽华尔兹——单调队列优化dp
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1499 简单的单调队列优化dp.(然而当时却WA得不行.今天总算填了坑) 注意滚动数组赋初值应 ...
- 单调队列优化DP || [NOI2005]瑰丽华尔兹 || BZOJ 1499 || Luogu P2254
题外话:题目极好,做题体验极差 题面:[NOI2005]瑰丽华尔兹 题解: F[t][i][j]表示第t时刻钢琴位于(i,j)时的最大路程F[t][i][j]=max(F[t-1][i][j],F[t ...
- P4381 [IOI2008]Island(基环树+单调队列优化dp)
P4381 [IOI2008]Island 题意:求图中所有基环树的直径和 我们对每棵基环树分别计算答案. 首先我们先bfs找环(dfs易爆栈) 蓝后我们处理直径 直径不在环上,就在环上某点的子树上 ...
- 【笔记篇】单调队列优化dp学习笔记&&luogu2569_bzoj1855股票交♂易
DP颂 DP之神 圣洁美丽 算法光芒照大地 我们怀着 崇高敬意 跪倒在DP神殿里 你的复杂 能让蒟蒻 试图入门却放弃 在你光辉 照耀下面 AC真心不容易 dp大概是最经久不衰 亘古不化的算法了吧. 而 ...
- 单调队列优化DP,多重背包
单调队列优化DP:http://www.cnblogs.com/ka200812/archive/2012/07/11/2585950.html 单调队列优化多重背包:http://blog.csdn ...
随机推荐
- 使用 Select2 下拉框实现复选
使用 Select2 下拉框实现复选 <!DOCTYPE html> <html> <head> <meta http-equiv="Content ...
- Delphi 文件转换Base64
uses EncdDecd; function FileToBase64(FileName: string): string; var MemoryStream: TMemoryStream;beg ...
- 排序算法五:随机化快速排序(Randomized quicksort)
上一篇提到,快速排序的平均时间复杂度是O(nlgn),比其他相同时间复杂度的堆排序.归并排序都要快,但这是有前提的,就是假定要排序的序列是随机分布的,而不是有序的.实际上,对于已经排好的序列,如果用快 ...
- 《STL源码剖析》——第一、二、三章
第一章:概论: 换句话说,STL所实现的,是依据泛型思维架设起来的一个概念结构.这个以抽象概念(abstract concepts)为主体而非以实际类(classes)为主体的结构,形成了一个严谨的 ...
- ELK日志分析系统之logstash7.x最新版安装与配置
2 .Logstash的简介 2.1 logstash 介绍 LogStash由JRuby语言编写,基于消息(message-based)的简单架构,并运行在Java虚拟机(JVM)上.不同于分离的代 ...
- 《JAVA设计模式》之策略模式(Strategy)
在阎宏博士的<JAVA与模式>一书中开头是这样描述策略(Strategy)模式的: 策略模式属于对象的行为模式.其用意是针对一组算法,将每一个算法封装到具有共同接口的独立的类中,从而使得它 ...
- RedHat可用的几处软件源
rpmforge仓库 http://repoforge.org/use/ http://rpms.famillecollet.com/
- 从0 开始手写一个 RPC 框架,轻松搞定!
Java技术栈 www.javastack.cn 优秀的Java技术公众号 来源:juejin.im/post/5c4481a4f265da613438aec3 之前在 RPC框架底层到底什么原理得知 ...
- 如何优化MySQL千万级大表
很好的一篇博客,转载 如何优化MySQL千万级大表 原文链接::https://blog.csdn.net/yangjianrong1985/article/details/102675334 千万级 ...
- elasticsearch 基础 —— Field Collapsing字段折叠
允许根据字段值折叠搜索结果.通过按折叠键选择顶部排序文档来完成折叠.例如,下面的查询检索每个用户的最佳推文,并按喜欢的数量对它们进行排序. GET /twitter/_search { "q ...