【BZOJ2806】【CTSC2012】Cheat 广义后缀自动机+二分+Dp
题目
思路&做法
我们先对标准作文库建广义后缀自动机。
然后对于每一篇阿米巴的作文, 我们首先把放到广义后缀自动机跑一遍, 对于每一个位置, 记录公共子串的长度\((\)即代码和下文中的\(val\)数组\()\)
接着我们二分答案, 用DP检验。
Dp方程很好想, \(d_i = max \{ d_j + i - j \ | \ i-val_i <= j <= i-lim \}\)
可以用单点队列优化。
代码
#include <iostream>
#include <cstdio>
#include <cstring>
#include <cstdlib>
using namespace std;
const int N = 1300010; //数组不能太大, 会T的
int n, m;
int val[N];
struct Suffix_Automaton
{ int nxt[N][2], fail[N], sz;
int len[N];
int root;
Suffix_Automaton() { }
inline int newnode(int l)
{ memset(nxt[sz], 0, sizeof(nxt[sz]));
fail[sz] = 0;
len[sz] = l;
return sz++;
}
void init()
{ sz = 1;
root = newnode(0);
}
inline int idx(char x) { return x - '0'; }
int add(int last, char x)
{ int c = idx(x);
if (nxt[last][c])
{ int p = last, q = nxt[last][c];
if (len[q] == len[p] + 1)
return q;
else
{ int u = newnode(len[p] + 1);
for (int i = 0; i < 2; i++) nxt[u][i] = nxt[q][i];
fail[u] = fail[q];
fail[q] = u;
while (p && nxt[p][c] == q)
{ nxt[p][c] = u;
p = fail[p];
}
return u;
}
}
else
{ int now = newnode(len[last] + 1);
int p = last;
while (p && !nxt[p][c])
{ nxt[p][c] = now;
p = fail[p];
}
if (!p) fail[now] = root;
else
{ int q = nxt[p][c];
if (len[q] == len[p] + 1)
fail[now] = q;
else
{ int u = newnode(len[p] + 1);
for (int i = 0; i < 2; i++) nxt[u][i] = nxt[q][i];
fail[u] = fail[q];
fail[now] = fail[q] = u;
while (p && nxt[p][c] == q)
{ nxt[p][c] = u;
p = fail[p];
}
}
}
return now;
}
}
void insert(char *s)
{ int Len = strlen(s);
int last = root;
for (int i = 0; i < Len; i++)
last = add(last, s[i]);
}
void work(char *str)
{ int cnt = 0;
int now = root;
int Len = strlen(str+1);
for (int i = 1; i <= Len; i++)
{ int c = idx(str[i]);
if (nxt[now][c])
{ cnt++;
now = nxt[now][c];
}
else
{ while (now && !nxt[now][c]) now = fail[now];
if (!now) { now = root; cnt = 0; }
else { cnt = len[now] + 1; now = nxt[now][c]; }
}
val[i] = cnt;
}
}
} tzw;
int d[N];
int Q[N], hd, tl;
bool check(char *s, int lim)
{ int Len = strlen(s+1);
for (int i = 0; i <= lim; i++) d[i] = 0;
hd = 0, tl = 1;
for (register int i = lim; i <= Len; i++)
{ d[i] = d[i-1];
if (i > lim) //这里一定不能去掉, 去掉会RE
{ while (hd < tl && Q[hd] < i-val[i]) hd++;
while (hd < tl && d[Q[tl-1]]+i-Q[tl-1] < d[i-lim]+i-(i-lim)) tl--;
Q[tl++] = i - lim;
}
if (val[i] >= lim) d[i] = max(d[i], d[Q[hd]] + i - Q[hd]);
}
return 10*d[Len] >= 9*Len;
}
int solve(char *s)
{ int l = 1, r = strlen(s+1);
while (l <= r)
{ int mid = (l+r) >> 1;
if (check(s, mid)) l = mid + 1;
else r = mid - 1;
}
return r;
}
char str[N];
int main()
{ scanf("%d %d", &n, &m);
tzw.init();
for (register int i = 1; i <= m; i++)
{ scanf("%s", str);
tzw.insert(str);
}
for (register int i = 1; i <= n; i++)
{ scanf("%s", str+1);
tzw.work(str);
if(!check(str, 1)) puts("0");
else printf("%d\n", solve(str));
}
return 0;
}
备注
注释里的坑我全踩了
【BZOJ2806】【CTSC2012】Cheat 广义后缀自动机+二分+Dp的更多相关文章
- 【BZOJ2806】[Ctsc2012]Cheat 广义后缀自动机+二分+单调队列优化DP
[BZOJ2806][Ctsc2012]Cheat Description Input 第一行两个整数N,M表示待检查的作文数量,和小强的标准作文库的行数接下来M行的01串,表示标准作文库接下来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】
题目 输入格式 第一行两个整数N,M表示待检查的作文数量,和小强的标准作文库 的行数 接下来M行的01串,表示标准作文库 接下来N行的01串,表示N篇作文 输出格式 N行,每行一个整数,表示这篇作文的 ...
- BZOJ 2806: [Ctsc2012]Cheat [广义后缀自动机 单调队列优化DP 二分]
2806: [Ctsc2012]Cheat 题意: 多个主串和多个询问串,每次询问将询问串分成多个连续子串,如果一个子串长度>=L且在主串中出现过就是熟悉的 如果熟悉的字符串长度>=询问串 ...
- bzoj 2806 [Ctsc2012]Cheat——广义后缀自动机+单调队列优化DP
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2806 只想着怎么用后缀数据结构做,其实应该考虑结合其他算法. 可以二分那个长度 L .设当前 ...
- BZOJ.2806.[CTSC2012]Cheat(广义后缀自动机 DP 单调队列)
题目链接 首先二分答案L.然后就是判断能否将原串划分出一些长度不小于L的子串,这些子串要是给定n个串中的某个串的子串,且满足它们的长度之和不小于原串长度的90%. 贪心多长选一段什么的显然不对.老老实 ...
- 【BZOJ2806】Cheat(后缀自动机,二分答案,动态规划,单调队列)
[BZOJ2806]Cheat(后缀自动机,二分答案,动态规划,单调队列) 题面 BZOJ 洛谷 题解 很有趣的一道题啊 对于在所有的串上面进行匹配? 很明显的后缀自动机 所以先构建出广义后缀自动机 ...
- [CTSC2012]熟悉的文章(广义后缀自动机+二分答案+单调队列优化DP)
我们对作文库建出广义后缀自动机.考虑用\(SAM\)处理出来一个数组\(mx[i]\),表示从作文的第\(i\)个位置向左最远在作文库中出现的子串的长度.这个东西可以在\(SAM\)上跑\(trans ...
- 【BZOJ2806】【CTSC2012】Cheat - 广义后缀自动机+单调队列优化DP
题意: Description Input 第一行两个整数N,M表示待检查的作文数量,和小强的标准作文库的行数 接下来M行的01串,表示标准作文库 接下来N行的01串,表示N篇作文 Output N行 ...
随机推荐
- AO如何获取SDE数据库中的数据
/// <summary> /// 获取Table类型表的记录 /// </summary> /// <param name="relationCheckCla ...
- 【转】linux命令
shell实例手册 0 说明{ 手册制作: 雪松 更新日期: 2015-11-02 欢迎系统运维加入Q群: 198173206 # 加群请回答问题 欢迎运维开发加入Q群: 3655344 ...
- ANE打包
哈哈,曾经梦寐以求的ANE终于弄成功了一个.说实话,学java和Android就是为了写ANE!好啦,今天把我体会到的记录一下: 网上其实打包ANE的教程好多,我也找了好多好多.但是好多我自己试了还是 ...
- sessionStorage和localStorage存储的转换不了json
先说说localStorage与sessionStorage的差别 sessionStorage是存储浏览器的暂时性的数据,当关闭浏览器下次再打开的时候就不能拿到之前存储的缓存了 localStora ...
- log日志模块得作用
import logginglogger=logging.getLogger()logger.setLevel(logging.DEBUG)#控制台输出日志'''consle=logging.Stre ...
- typora与Markdown的一些小问题
一.typora中修改图像大小 加上style="zoom:50%" <img src="E:\GitHub_learn\blog\source\imgs\tree ...
- 【剑指Offer】6、旋转数组的最小数字
题目描述: 把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转. 输入一个非减排序的数组的一个旋转,输出旋转数组的最小元素. 例如数组{3,4,5,1,2}为{1,2,3,4,5 ...
- 被遗忘的 Logrotate
转自: http://huoding.com/2013/04/21/246 被遗忘的 Logrotate 发表于 2013-04-21 我发现很多人的服务器上都运行着一些诸如每天切分 Nginx 日志 ...
- js-2018-11-09 关于Array中的srot()方法和compare()方法
Array中的srot()方法 sort()方法是用来重排序的方法.在默认情况下,sort()方法按升序排列数组项----即最小的值位于最前面,最大的值排在最后面. 我们看看官方是怎么说的: arra ...
- Monkey基本常用命令整理
adb shell monkey -v 500 >F:/monkeylog2018.txt -P表示包名 -V表示输出日志的详细级别 一个-V表示一级 递增 >输出日志 ...