Ctsc2012的题目。做完感觉自己瞬间变高富帅了。

不过回想其实也觉得不难,想到用单调队列就很简单了,还有二分= =。呵

对于给出的一篇文章,如果你们将它分成若干段,并在所有长度不小于L的片段在字典中间出现的总长度和不小于原文的90%,那么这篇文章就可以被认为是熟悉的。这里要注意理解一下题意,如果你要算对于L这个长度熟悉。那么小于L的片段就不要去判断了,它肯定不熟悉。

这样我们就可以二分了,每次二分一个长度,看看能否使文章熟悉就好,最终输出答案。

怎么判断当前这个二分的答案能否使得文章熟悉呢?

其实这个问题,就是求把文章分段后的最大的匹配长度。这样由于存在策略最优的问题,而且显然可以递推,就是DP了。

如何DP呢?f[i]表示前面i个字符的最大匹配长度。对于当前我们考虑的状态,我们只要考虑它为某一段尾部最后一个字符就可以了。

同时后缀自动机可以在线秒算当前位置的最大匹配长度,假设i位置的最大匹配长度为len。

首先,f[i]=f[i-1],因为i个位置的匹配长度肯定不会小于前一个位置的匹配长度。

其次,如果当前位置的匹配长度小于当前枚举的这个L值的话,当前状态也是不予考虑的。因为即使当做尾部最后一个字符也不算数。

再其次,当前状态的决策区间只可能是[i-len,i-L]。因为小于i-len的位置是不能匹配的。

再其次,每次进入一个点,我们都进行队尾的优化,可以优化么?可以。根据前面总共有多少个位置匹配不上来优化,如果一个前面的位置,落下的无法匹配的字符比后面一个位置的空字符还要多,那么它肯定不会在后面的决策总使用,直接从队尾拿出来就好了。

再其次,队首的优化就是判断队首的点时候在决策区间内,如果在的话,当前状态就一定是最优的了。也就是f[j]+i-j。因为在决策区间里的位置,后面的i-j的长度都是可以完美当做一段去匹配的。

再其次,没有了,直接判断是否满足90%的条件即可。

怒赞,ctsc题目果然做起来感觉不一样。

召唤代码君:

#include <iostream>
#include <cstdio>
#include <cstring>
#define maxn 2502000
using namespace std; int next[maxn][3],pre[maxn],step[maxn];
int p,q,np,nq,cur,len,L;
int f[maxn];
int N,last,n,m;
int Q[maxn],bot,top;
char s[maxn]; void insert(int x)
{
np=++N,p=last,step[np]=step[p]+1,last=np;
for (; p!=-1 && next[p][x]==0; p=pre[p]) next[p][x]=np;
if (p==-1) return;
q=next[p][x];
if (step[q]==step[p]+1) { pre[np]=q; return; }
nq=++N,step[nq]=step[p]+1,pre[nq]=pre[q];
for (int i=0; i<3; i++) next[nq][i]=next[q][i];
pre[np]=pre[q]=nq;
for (; p!=-1 && next[p][x]==q; p=pre[p]) next[p][x]=nq;
} bool check(int limit)
{
bot=1,top=0,f[0]=cur=len=0;
for (int i=1; s[i]; i++)
{
f[i]=f[i-1];
int k=s[i]-'0';
for (; cur!=-1 && next[cur][k]==0; cur=pre[cur]) ;
if (cur==-1) { cur=len=0; }
len=min(len,step[cur])+1;//当前状态的最大匹配长度
cur=next[cur][k]; p=i-limit;
if (p>=0)
{
while (bot<=top && Q[top]-f[Q[top]]>p-f[p]) top--;
Q[++top]=p;
}
while (bot<=top && Q[bot]<i-len) bot++;
if (bot<=top) f[i]=max(f[i],f[Q[bot]]+i-Q[bot]); }
return f[L]*10>=L*9;
} int main()
{
pre[0]=-1;
scanf("%d%d",&n,&m);
while (m--)
{
scanf("%s",s);
for (int i=0; s[i]; i++) insert(s[i]-'0');
insert(2);
}
while (n--)
{
scanf("%s",s+1);
L=strlen(s+1);
int l=0,r=L,mid;
while (l<r)
{
mid=(l+r+1)/2;
if (check(mid)) l=mid;
else r=mid-1;
}
printf("%d\n",l);
}
return 0;
}

  

BZOJ2806_Cheat的更多相关文章

随机推荐

  1. [Vue warn]:vue-Failed to resolve directive: clipboard

    前言 需求:移动端需要一个按钮,复制到剪切板,分享给好友(没有调用微信内置的分享接口) 插件 vue-clipboard2 环境:vue,node 安装:npm install --save vue- ...

  2. ASP.NET MVC Bundles 之学习笔记

    在网页中,我们经常需要引用大量的javascript和css文件,在加上许多javascript库都包含debug版和经过压缩的release版(比如jquery),不仅麻烦还很容易引起混乱,所以AS ...

  3. .net 控件生命周期

    这里列举出来了11个生命周期,一般的控件生命周期会经历这11个生命周期,但是有一些特别的控件比如页面控件System.Web.UI.Page等. 具体代码参考如下: /// <summary&g ...

  4. 曾经的华为C面试题,一点就通

     学习编程可以锻炼你的思维,帮助你更好地思考,创建一种我认为在各领域都非常有用的思维方式.   比尔盖茨      曾经的华为C面试题,一点就通 [问题区] 有两个变量x和y, x=10; y = 2 ...

  5. 【Ansible】ansible 任务失败控制

    任务失败控制 Ansible 通常默认会确保检测模块和命令的返回码并且会快速失败 – 专注于一个错误除非你另作打算. 有时一条命令会返回 0 但那不是报错.有时命令不会总是报告它 ‘改变’ 了远程系统 ...

  6. 学习笔记 | treap | splay

    目录 前言 treap 它的基本操作 前言 不会数据结构选手深深地感受到了来自treap的恶意QwQ 在听的时候感觉自己听得听懂的??大概只是听懂了它的意思 代码是怎么写都感觉写不好╮(╯﹏╰)╭ 菜 ...

  7. printf命令详解

    基础命令学习目录首页 本文是Linux Shell系列教程的第(八)篇,更多shell教程请看:Linux Shell系列教程 在上一篇:Linux Shell系列教程之(七)Shell输出这篇文章中 ...

  8. Currency Exchange 货币兑换 Bellman-Ford SPFA 判正权回路

    Description Several currency exchange points are working in our city. Let us suppose that each point ...

  9. js中模拟a标签的点击事件

    var a = document.createElement('a'); a.target = "_blank"; a.href = "personal"; a ...

  10. Daily Srum 10.21

    到目前为止,我们组处在学习阶段,很多知识点都还不太清楚,所以现在我们还在看相关书籍和博客,任务. 而我们此间主要是在阅读一些材料: 陈谋一直在看学长的代码,其中C#的很多方式我都不太明白(尽管和Jav ...