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 ... 
随机推荐
- Redis记录-JAVA连接Redis
			在Java程序中使用Redis之前,需要确保在机器上安装了Redis的Java驱动程序和Java环境.可以先在将Java电脑上并配置好环境. 安装 现在,让我们看看如何设置Redis Java驱动程序 ... 
- Java 调用 groovy 脚本文件,groovy 访问 MongoDB
			groovy 访问 MongoDB 示例: shell.groovy package db import com.gmongo.GMongoClient import com.mongodb.Basi ... 
- Codeforces Round #481 (Div. 3)  D. Almost Arithmetic Progression
			http://codeforces.com/contest/978/problem/D 题目大意: 给你一个长度为n的b(i)数组,你有如下操作: 对数组中的某个元素+1,+0,-1.并且这个元素只能 ... 
- JavaScript模拟QQ签名(HTML5 contenteditable属性)
			例图: 一.思路 1.单击元素时,使元素可以编辑,并获得焦点 2.按下键盘检测用户编辑元素中的文本 3.监听按下Enter键操作或离开可编辑元素焦点时,更新数据库 二.代码 $(function(){ ... 
- 兼容 IE6+ 获取图片大小
			昨天说了 HTML5 的file对象可以获取到文件的 文件名,文件大小,文件类型,最后一次修改日期.其实 IE6-9 也可以变向的获取到,虽然没那么方便,但至少可以取到.来看例子吧: <!doc ... 
- 浅谈tomcat中间件的优化【转】
			今天来总结一下tomcat的一些优化的方案,由于本人才疏学浅,写的不好,勿喷! tomcat对于大多数从事开发工作的童鞋应该不会很陌生,通常做为默认的开发环境来为大家服务,不过tomcat默认的一些配 ... 
- mac  安装gevent报错
			运行pip install gevent报错 错误信息如下 xcrun: error: invalid active developer path (/Library/Developer/Comman ... 
- c#调用c++ dll  入坑记录
			1.DLL引用坑 [DllImport("NetDLL.dll", CharSet = CharSet.Ansi, CallingConvention = CallingConve ... 
- Springboot分模块开发详解(2):建立子工程
			1.创建base-entity 选中base工程,右键创建一个新的maven工程 自动选择了base这个目录存放子工程 创建后,pom.xml修改成如下内容: <?xml version=&qu ... 
- ICCV2013 录用论文(目标跟踪相关部分)(转)
			单目标(表观模型): 1. Seunghoon Hong, BohyungHan. Orderless Trackingthrough Model-Averaged Density Estimatio ... 
