BZOJ.2806.[CTSC2012]Cheat(广义后缀自动机 DP 单调队列)
首先二分答案L。然后就是判断能否将原串划分出一些长度不小于L的子串,这些子串要是给定n个串中的某个串的子串,且满足它们的长度之和不小于原串长度的90%。
贪心多长选一段什么的显然不对。老老实实DP。
设\(f[i]\)为到\(i\)划分出的最长长度(不用想什么奇奇怪怪的状态啊→_→),则\(f[i]=\max\{f[i-1],\ f[j]+i-j\}\) (\(s[i\sim j]\)是\(n\)个串中某串的子串,且\(i-j\geq L\))。
求以某位置结尾的子串是否匹配,可以对\(n\)个串建广义SAM,原串在上面匹配就能得到每个位置作为后缀所能匹配的最大长度,记为\(mx[i]\)。
那么\(j\)的范围就是\(i-mx[i]\leq j\leq i-L\).
这是\(n^2\)的,要优化。因为贪心不对,区间内的数还是要都尝试更新一遍的。观察决策位置是否有单调性,比如\(i\)与\(i+1\),有\(mx[i]+1\geq mx[i+1]\)。
即决策位置是单调不降的。只需用单调队列维护当前区间$f[j]-j$的最值就可以了。
另外卡精度,0.9*n会偏大?要减个eps。(不想再看浮点数怎么存储了...太sxbk了吧)
顺便题目描述挺有趣233
```cpp
//63128kb 820ms
#include <cstdio>
#include <cctype>
#include <cstring>
#include <algorithm>
#define gc() getchar()
#define eps 1e-8
const int N=2200007;//字节。。
int n,m;
struct Suffix_Automaton
{
int tot,las,son[N][2],fa[N],len[N],mx[N],q[N],f[N];
char s[N];
Suffix_Automaton() {tot=las=1;}
void Insert(int c)
{
int p=las,np=++tot; len[las=np]=len[p]+1;
for(; p&&!son[p][c]; p=fa[p]) son[p][c]=np;
if(!p) fa[np]=1;
else
{
int q=son[p][c];
if(len[q]==len[p]+1) fa[np]=q;
else
{
int nq=++tot; len[nq]=len[p]+1;
memcpy(son[nq],son[q],sizeof son[q]);
fa[nq]=fa[q], fa[q]=fa[np]=nq;
for(; son[p][c]==q; p=fa[p]) son[p][c]=nq;
}
}
}
void Build()
{
scanf("%s",s), las=1;//!
for(int i=0,l=strlen(s); i<l; ++i) Insert(s[i]-'0');
}
void Get_mx(char *s)
{
for(int c,now=0,p=1,i=1,l=strlen(s+1); i<=l; mx[i++]=now)
if(son[p][c=s[i]-'0']) ++now, p=son[p][c];
else
{
for(; p&&!son[p][c]; p=fa[p]);
if(!p) now=0, p=1;
else now=len[p]+1, p=son[p][c];
}
}
bool Check(int L,int n)
{
int h=1,t=0; f[0]=0;
for(int i=1; i<=n; ++i)
{
f[i]=f[i-1];
if(i>=L && L<=mx[i])
{
int p=i-L;
while(h<=t && f[q[t]]-q[t]<=f[p]-p) --t;
q[++t]=p;
}
while(h<=t && q[h]<i-mx[i]) ++h;
if(h<=t) f[i]=std::max(f[i],i+f[q[h]]-q[h]);//好像f[0]=INF不太方便
}
return (double)f[n]>=0.9*n-eps;//0.89999999
}
void Query()
{
scanf("%s",s+1), Get_mx(s);
int len=strlen(s+1);
int l=1,r=len,mid,ans=0;
while(l<=r)
if(Check(mid=l+r>>1,len)) ans=mid,l=mid+1;
else r=mid-1;
printf("%d\n",ans);
}
}sam;
int main()
{
scanf("%d%d",&n,&m);
while(m--) sam.Build();
while(n--) sam.Query();
return 0;
}
```\]
BZOJ.2806.[CTSC2012]Cheat(广义后缀自动机 DP 单调队列)的更多相关文章
- BZOJ 2806: [Ctsc2012]Cheat [广义后缀自动机 单调队列优化DP 二分]
2806: [Ctsc2012]Cheat 题意: 多个主串和多个询问串,每次询问将询问串分成多个连续子串,如果一个子串长度>=L且在主串中出现过就是熟悉的 如果熟悉的字符串长度>=询问串 ...
- bzoj 2806 [Ctsc2012]Cheat——广义后缀自动机+单调队列优化DP
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2806 只想着怎么用后缀数据结构做,其实应该考虑结合其他算法. 可以二分那个长度 L .设当前 ...
- 【BZOJ2806】[Ctsc2012]Cheat 广义后缀自动机+二分+单调队列优化DP
[BZOJ2806][Ctsc2012]Cheat Description Input 第一行两个整数N,M表示待检查的作文数量,和小强的标准作文库的行数接下来M行的01串,表示标准作文库接下来N行的 ...
- 【BZOJ 2806】 2806: [Ctsc2012]Cheat (SAM+二分+DP+单调队列)
2806: [Ctsc2012]Cheat Time Limit: 20 Sec Memory Limit: 256 MBSubmit: 1262 Solved: 643 Description ...
- BZOJ2806 [Ctsc2012]Cheat 【后缀自动机 + 二分 + 单调队列优化DP】
题目 输入格式 第一行两个整数N,M表示待检查的作文数量,和小强的标准作文库 的行数 接下来M行的01串,表示标准作文库 接下来N行的01串,表示N篇作文 输出格式 N行,每行一个整数,表示这篇作文的 ...
- BZOJ 2806 Luogu P4022 [CTSC2012]Cheat (广义后缀自动机、DP、二分、单调队列)
题目链接: (bzoj) https://www.lydsy.com/JudgeOnline/problem.php?id=2806 (luogu) https://www.luogu.org/pro ...
- BZOJ2806: [Ctsc2012]Cheat(广义后缀自动机,单调队列优化Dp)
Description Input 第一行两个整数N,M表示待检查的作文数量,和小强的标准作文库的行数接下来M行的01串,表示标准作文库接下来N行的01串,表示N篇作文 Output N行,每行一个整 ...
- CTSC2012 熟悉的文章 广义后缀自动机_单调队列
没啥难的,主要是单调队列忘了咋求了QAQ... Code: #include <cstdio> #include <algorithm> #include <cstrin ...
- bzoj 2806: [Ctsc2012]Cheat 后缀自动机DP
2806: [Ctsc2012]Cheat Time Limit: 20 Sec Memory Limit: 256 MBSubmit: 583 Solved: 330[Submit][Statu ...
随机推荐
- Intellij-idea 如何编译maven工程
小编最近效应项目的要求,学习在idea上编写项目.作为一个新手遇到问题也算是正常的,重要的是把它解决,get新技能. 编写过maven工程的小伙伴们应该都知道怎么在eclipse中编译maven工程: ...
- bzoj千题计划187:bzoj1770: [Usaco2009 Nov]lights 燈 (高斯消元解异或方程组+枚举自由元)
http://www.lydsy.com/JudgeOnline/problem.php?id=1770 a[i][j] 表示i对j有影响 高斯消元解异或方程组 然后dfs枚举自由元确定最优解 #in ...
- win10内建子系统Linux
从cmd中下载linux和linux终端安装程序一样 最新要从商店购买(当然是免费) ubuntu openSUSE SUSE 3个 创建用户
- Linux(Debian)软件安装
# 配置/etc/apt/sources.list 通过root权限修改/etc/apt/sources.list $ su #输入密码进入root权限 $ chmod 0666 /etc/apt/s ...
- Markdown 详细语法
<< 访问 Wow!Ubuntu NOTE: This is Simplelified Chinese Edition Document of Markdown Syntax. If yo ...
- 跳过复制错误——slave_skip_errors、slave_exec_mode
这一篇写写复制错误处理相关的另两个参数slave_skip_errors.slave_exec_mode,基本环境参考<复制错误处理——sql_slave_skip_counter> 一. ...
- readb(), readw(), readl(),writeb(), writew(), writel() 宏函数【转】
转自:http://www.netfoucs.com/article/hustyangju/70429.html readb(), readw(), readl()函数功能:从内存映射的 I/O 空间 ...
- Daemon函数的用法
Daemon函数的用法 说明: 让一个程序后台运行. 原型: #include <unistd.h> int daemon(int nochdir, int noclose); #incl ...
- Redis常见业务场景应用
一定时间范围内不可重复发短信问题 Redis实现消息队列 Redis实现Session共享 ...
- Nuxt学习笔记
参考地址:https://zh.nuxtjs.org/guide/installation 官网 http://jspang.com/2018/02/26/nuxt/ 1.目录结构 2.Nuxt常用 ...