bzoj 2806: [Ctsc2012]Cheat 后缀自动机DP
2806: [Ctsc2012]Cheat
Time Limit: 20 Sec Memory Limit: 256 MB
Submit: 583 Solved: 330
[Submit][Status][Discuss]
Description


Input
第一行两个整数N,M表示待检查的作文数量,和小强的标准作文库
的行数
接下来M行的01串,表示标准作文库
接下来N行的01串,表示N篇作文
Output
N行,每行一个整数,表示这篇作文的Lo 值。
Sample Input
10110
000001110
1011001100
Sample Output
HINT
输入文件不超过1100000字节
注意:题目有改动,可识别的长度不小于90%即可,而不是大于90%
先将作文库建后缀自动机,多篇文章可以通过在中间加入分隔符来完成。对于每组询问,预处理出每一个位置向前最多匹配多长g[]。
二分答案len,dp[]表示匹配到当前位置的最多匹配数,对于i位置,dp[i]由一段通过len,与g确定出的区间[l,r]转移,本来用了一个线段树维护,但是由于时间复杂度O(n*log^2n),TLE了,观察发现[l,r]是单调的,故可直接用单调队列。
网上一半题解过不了数据:1 1 1 1
另外,对于0.9的问题确实说明了以后能用int就不要用double,实在不行要加eps。
省选前最后一题了,真觉得时间过得太快了。
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
#define MAXN 4100000
#define MAXT MAXN*4
#define INF 0x3f3f3f3f
#define lch (now<<1)
#define rch (now<<1^1)
#define smid ((l+r)>>1)
char buf[MAXN];
char *bufnow(buf);
int len[MAXN];
char *str[MAXN];
char buf2[MAXN];
char *bufnow2(buf2);
struct sam_node
{
int nxt[];
int pnt,len;
void Print()
{
for (int i=;i<;i++)
printf("%d[%d] ",i,nxt[i]);
printf("\n");
printf("Pnt:%d\n",pnt);
}
}sam[MAXN];
int tops=;
int last=;
void Add_item(int w)
{
int p=last;
int np=++tops;
sam[np].len=sam[p].len+;
while (p && !sam[p].nxt[w])
sam[p].nxt[w]=np,p=sam[p].pnt;
if (!p)
{
last=np;
sam[np].pnt=;
}else
{
int q=sam[p].nxt[w];
if (sam[p].len+==sam[q].len)
{
sam[np].pnt=q;
}else
{
int nq=++tops;
sam[nq]=sam[q];
sam[nq].len=sam[p].len+;
sam[nq].pnt=sam[q].pnt;
sam[q].pnt=nq;
sam[np].pnt=nq;
while (p && sam[p].nxt[w]==q)
{
sam[p].nxt[w]=nq;
p=sam[p].pnt;
}
}
}
last=np;
}
int g[MAXN];
int dp[MAXN];
int seq[MAXN]; int main()
{
freopen("input.txt","r",stdin);
int n,m;
int x,y;
scanf("%d%d\n",&n,&m);
for (int i=;i<m;i++)
{
scanf("%s\n",bufnow2);
bufnow2+=strlen(bufnow2);
*(bufnow2++)='';
}
for (int i=;i<n;i++)
{
scanf("%s\n",bufnow);
str[i]=bufnow;
bufnow+=len[i]=strlen(bufnow);
bufnow++;
}
for (char *i=buf2;i!=bufnow2;i++)
{
Add_item(*i-'');
}
for (int i=;i<=tops;i++)
{
// printf("SAM<%d>:\n",i);
// sam[i].Print();
}
for (int i=;i<n;i++)
{
int now=;
int clen=;
for (int j=;j<len[i];j++)
{
int w=str[i][j]-'';
if (sam[now].nxt[w])
{
now=sam[now].nxt[w];
clen++;
}else
{
while (now && !sam[now].nxt[w])
now=sam[now].pnt;
if (!now)
{
now=;
clen=;
}else
{
clen=sam[now].len+;
now=sam[now].nxt[w];
}
}
g[j]=clen;
// printf("%d\n",clen);
}
for (int j=len[i];j>=;j--)
g[j]=g[j-];
g[]=;
int l=,r=len[i]+;
int mid;
int head,tail=;
int ny;
while (l+<r)
{
mid=(l+r)>>;
int t;
for (int j=;j<=len[i];j++)
dp[j]=-INF;
dp[]=;
head=,tail=-;
ny=;
for (int j=;j<=len[i];j++)
{
x=j-g[j];
y=j-mid;
while (ny<=y)
{
while (head<=tail && dp[seq[tail]]<=dp[ny])
tail--;
seq[++tail]=ny++;
}
while (head<=tail && seq[head]<x)
head++;
dp[j]=dp[j-]+(j-);
if (head<=tail)
dp[j]=max(dp[j],dp[seq[head]]+j);
dp[j]-=j;
}
t=dp[len[i]]+len[i];
if (t*>=len[i]*)
l=mid;
else
r=mid;
}
printf("%d\n",l);
}
}
bzoj 2806: [Ctsc2012]Cheat 后缀自动机DP的更多相关文章
- BZOJ 2806 [Ctsc2012]Cheat ——后缀自动机 单调队列优化DP
先建出广义后缀自动机. 然后跑出文章中每一个位置的最大匹配距离. 然后定义$f[i]$表示匹配到以$i$结尾的串时,最长的匹配距离. 显然可以二分$L$的取值. 然后容易得到$DP$方程 $f[i]= ...
- BZOJ 2806 [Ctsc2012]Cheat (后缀自动机+二分+单调队列+dp)
题目大意: 给你一堆模式串和文本串 对于每个文本串,我们可以把它不可重叠地拆分成很多子串,如果拆分出的串作为子串出现在了任何一个模式串中,我们称它是“眼熟的”,我们必须保证“眼熟的”子串总长度不小于文 ...
- BZOJ 2806: [Ctsc2012]Cheat [广义后缀自动机 单调队列优化DP 二分]
2806: [Ctsc2012]Cheat 题意: 多个主串和多个询问串,每次询问将询问串分成多个连续子串,如果一个子串长度>=L且在主串中出现过就是熟悉的 如果熟悉的字符串长度>=询问串 ...
- [bzoj2806][Ctsc2012]Cheat(后缀自动机(SAM)+二分答案+单调队列优化dp)
偷懒直接把bzoj的网页内容ctrlcv过来了 2806: [Ctsc2012]Cheat Time Limit: 20 Sec Memory Limit: 256 MBSubmit: 1943 ...
- BZOJ 2806: [Ctsc2012]Cheat(单调队列优化dp+后缀自动机)
传送门 解题思路 肯定先要建出来广义后缀自动机.刚开始以为是个二分+贪心,写了一下结果\(20\)分.说一下正解,首先显然\(L_0\)具有单调性,是可以二分的.考虑二分后怎样判合法,对于分割序列很容 ...
- bzoj 2806 [Ctsc2012]Cheat——广义后缀自动机+单调队列优化DP
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2806 只想着怎么用后缀数据结构做,其实应该考虑结合其他算法. 可以二分那个长度 L .设当前 ...
- BZOJ.2806.[CTSC2012]Cheat(广义后缀自动机 DP 单调队列)
题目链接 首先二分答案L.然后就是判断能否将原串划分出一些长度不小于L的子串,这些子串要是给定n个串中的某个串的子串,且满足它们的长度之和不小于原串长度的90%. 贪心多长选一段什么的显然不对.老老实 ...
- bzoj 2806: [Ctsc2012]Cheat【广义SAM+二分+dp+单调队列】
把模板串建一个广义SAM 然后在线查询,每次在SAM上预处理出一个a[i]表示i位置向前最多能匹配多长的模板串 二分答案L,dp判断,设f[i]为·~i有几个匹配,转移显然是f[i]=max{f[i- ...
- bzoj 2806: [Ctsc2012]Cheat
传送门 好久没刷bzoj惹…… 题意不说可以嘛. 首先二分答案. SAM的事情搞完以后就是dp辣. 我们已经对于每个位置i,找到了最小的一个k,使得[k,i]这个子串在模版串中出现过.那么我们需要做的 ...
随机推荐
- 触摸与手势学习-swift
触摸是一个UITouch对象,该对象放在一个UIEvent中,每个UIEvent包含一个或多个UITouch对象,每个UITouch对象对应一个手指.系统将UIEvent发送到应用程序上,最后应用程序 ...
- js calendar橙色日期选择器代码
原文出处 http://files.cnblogs.com/files/quixon/date_js.rar
- MySQL的链接,查看数据库,使用数据库,查看表
MySQL的链接,查看数据库,使用数据库,查看表 mysql> show databases; +--------------------+ | Database | +------------ ...
- javaweb 中的乱码问题
一.post 方式 首先我们看下面一段代码,在该HTML中我们指定的编码为“UTF-8”,如图所示. 在该代码中,我们将表单数据提交给ParamServlet 处理 servlet 会将接收到的数据打 ...
- C#面向对象的一些东西
最近在复习C#面向对象,也就是说常说的3大特性:封装.继承和多态.首先说一下封装,其实封装最大的目的也是为了实现代码的解耦和重用.代码也是安全的(对外它隐藏了具体的实现,就好比我们拿个遥控器就能操作电 ...
- 从两个集合里排除重复的写法(适用:DB表和字段都很多,表间有关联的情况)
获取其中一张表bulletinred为1的内容: public IList<BRShow> GetBulInfo() { var result = from a in ((Entities ...
- 响应者链 hittest:withEvent: 方法的使用
关于响应者链部分的基础内容 参考http://www.cnblogs.com/wendingding/p/3795171.html 这里我要说明的是 关于- (UIView *)hitTest:(CG ...
- 暑假集训(1)第六弹 -----简单计算器(Hdoj1237)
Description 读入一个只包含 +, -, *, / 的非负整数计算表达式,计算该表达式的值. Input 测试输入包含若干测试用例,每个测试用例占一行,每行不超过200个字符,整数和运算 ...
- javascript 对象的创建,引用,释放,删除方法
1.用函数构造 A.声明时同时设置属性和方法 function func(){ this.name = "myname"; this.say = function(){aler ...
- C# WinForm给Button按钮或其它控件添加快捷键响应
就在这介绍三种添加快捷键的方式. 第一种Alt + *(按钮快捷键) 在大家给button.label.menuStrip等控件设置Text属性时在名字后边加&键名就可以了,比如button1 ...