「BZOJ 2534」 L - gap字符串

题目描述

有一种形如 \(uv u\) 形式的字符串,其中 \(u\) 是非空字符串,且 \(v\) 的长度正好为 \(L\), 那么称这个字符串为 \(L-Gap\) 字符串 给出一个字符串 \(S\), 以及一个正整数 \(L\), 问 \(S\) 中有多少个 \(L-Gap\) 子串.

\(1 \leq |S| \leq 5 \times 10^4, L \leq 10\)

### 解题思路 :

考虑要对特征串计数,不妨枚举 单个 \(u\) 的长度 \(|u|\),在串上设置 \(\left\lfloor \frac{|s|}{|u|}\right\rfloor\) 个关键点.

计算单个 \(u\) 覆盖某一关键点的所有串对应的 \(uvu\) 串的总数

考虑当前有关键点 \(P_1\) 其对应的另外一个 \(u\) 所在的左端点点为 \(P_2\)

考虑 \(P_1, P_2\) 能作为 \(uvu\) 串中两个\(u\)的左端点的话,当且仅当 \(Lcp(P_1, P_2) \geq |u|\)

那么问题就转化为,所有覆盖关键点 \(P_1\) 的 \(u\) 串,与其对应点的 \(Lcp \geq |u|\) 的个数是多少

考虑覆盖 \(P_1\) 的 \(u\) 串的左端点为 \(P_1'\) ,\((P_1 - |u| < P_1' < P_1 + |u|)\),其对应的另外一个\(u\) 的左端点为 \(P_2'\)

观察发现 \(P_1', P_2'\) 显然是由 \(P_1', P_2'\) 平移同一距离得到的

不妨求出后缀 \(P_1, P_2\) 的 \(Lcp\) 以及前缀 \(P_1, P_2\) 的 \(Lcs\),也就是图中红色和蓝色的线段

显然 \(P_1', P_2'\) 移动到线段之外后不可能满足 \(Lcp(P_1', P_2') \geq |u|\),所以答案就是线段上可行的 \(P_1'\) 数

也就是 \(\max(Lcp + Lcs - |u|, 0)\)

考虑这样做为什么是正确的

首先是时间复杂度,每次枚举一个 \(|u|\) ,对于 \(\left\lfloor \frac{|s|}{|u|}\right\rfloor\) 个关键点 \(O(1)\) 或 \(O(log|s|)\) 算 \(Lcs, Lcp\)

根据调和级数,复杂度是 \(O(|s|log|s|)\) 或 \(O(|s|log^2|s|)\),足以通过此题

接下来分析算法的正确性,也就是为什么不会算重和算漏

考虑对于每一个长度为 \(|u|\) 的 \(u\) 串,当且仅当只能覆盖一个关键点,所以如果其有答案,必然会被关键点算到且被算一次

注意 \(Lcp\) 和 \(Lcs\) 要对 \(|u|\) 取模,不然算到的答案就不是覆盖当且关键点的答案,会因为算法写挂了而算重

这里我用了二分 + \(hash\) 求 \(Lcs, Lcp\) ,是两个 \(log\) 的,实际上如果用 \(Sa\) 的话,复杂度就如上述一样,省去一个 \(log\)

/*program by mangoyang*/
#include<bits/stdc++.h>
#define inf (0x7f7f7f7f)
#define N (100005)
typedef long long ll;
typedef unsigned long long ull;
using namespace std;
const ull base = 233;
char s[N];
int n, m, ans;
ull pw[N], hs[N];
inline ull get(int l, int r){
if(l > r) return 0; return hs[r] - hs[l-1] * pw[r-l+1];
}
inline int getpre(int x, int y, int lim){
int l = 0, r = lim, ans = 0;
while(l <= r){
int mid = l + r >> 1;
if(get(x - mid + 1, x) == get(y - mid + 1, y))
ans = mid, l = mid + 1; else r = mid - 1;
}
return ans;
}
inline int getsuf(int x, int y, int lim){
int l = 0, r = lim, ans = 0;
while(l <= r){
int mid = l + r >> 1;
if(get(x, x + mid - 1) == get(y, y + mid - 1))
ans = mid, l = mid + 1; else r = mid - 1;
}
return ans;
}
inline void solve(int L, int R){
int len = (R - L + 1 - m) / 2, l1 = L, l2 = R - len + 1;
int llen = getpre(l1, l2, len), rlen = getsuf(l1, l2, len);
int now = (llen && rlen) ? llen + rlen - 1 : llen + rlen;
if(now >= len) ans += now - len + 1;
}
int main(){
pw[0] = 1, scanf("%d", &m);
scanf("%s", s + 1); int n = strlen(s + 1);
for(int i = 1; i < N; i++) pw[i] = pw[i-1] * base;
for(int i = 1; i <= n; i++) hs[i] = hs[i-1] * base + s[i];
for(int k = 1; k <= (n - m) / 2; k++)
for(int i = 1; i <= n; i += k) solve(i, i + 2 * k + m - 1);
cout << ans;
return 0;
}

「BZOJ 2534」 L - gap字符串的更多相关文章

  1. 「BZOJ 4502」串

    「BZOJ 4502」串 题目描述 兔子们在玩字符串的游戏.首先,它们拿出了一个字符串集合 \(S\),然后它们定义一个字符串为"好"的,当且仅当它可以被分成非空的两段,其中每一段 ...

  2. 「BZOJ 2956」模积和

    「BZOJ 2956」模积和 令 \(l=\min(n,m)\).这个 \(i\neq j\) 非常不优雅,所以我们考虑分开计算,即: \[\begin{aligned} &\sum_{i=1 ...

  3. 「BZOJ 4228」Tibbar的后花园

    「BZOJ 4228」Tibbar的后花园 Please contact lydsy2012@163.com! 警告 解题思路 可以证明最终的图中所有点的度数都 \(< 3\) ,且不存在环长是 ...

  4. 「BZOJ 3645」小朋友与二叉树

    「BZOJ 3645」小朋友与二叉树 解题思路 令 \(G(x)\) 为关于可选大小集合的生成函数,即 \[ G(x)=\sum[i\in c ] x^i \] 令 \(F(x)\) 第 \(n\) ...

  5. 「BZOJ 4289」 PA2012 Tax

    「BZOJ 4289」 PA2012 Tax 题目描述 给出一个 \(N\) 个点 \(M\) 条边的无向图,经过一个点的代价是进入和离开这个点的两条边的边权的较大值,求从起点 \(1\) 到点 \( ...

  6. Solution -「BZOJ 3812」主旋律

    \(\mathcal{Description}\)   Link.   给定含 \(n\) 个点 \(m\) 条边的简单有向图 \(G=(V,E)\),求 \(H=(V,E'\subseteq E)\ ...

  7. 「BZOJ 2434」「NOI 2011」阿狸的打字机「AC自动机」

    题意 有一个打字机,支持三种操作: 字符串末尾加一个小写字母 字符串末尾减一个字符 输出这个字符串 经过不超过\(n\)次操作后有\(m\)组询问:\((x,y)\),表示第\(x\)次输出第字符串在 ...

  8. 「BZOJ 1001」狼抓兔子

    题目链接 luogu bzoj \(Solution\) 这个貌似没有什么好讲的吧,直接按照这个给的图建图就好了啊,没有什么脑子,但是几点要注意的: 建双向边啊. 要这么写,中间还要写一个\(whil ...

  9. 「BZOJ 5188」「Usaco2018 Jan」MooTube

    题目链接 luogu bzoj \(Describe\) 有一个\(n\)个节点的树,边有权值,定义两个节点之间的距离为两点之间的路径上的最小边权 给你\(Q\)个询问,问你与点\(v\)的距离大于等 ...

随机推荐

  1. 【51NOD】1135 原根

    [题意]给定p,求p的原根g.3<=p<=10^9. [算法]数学 [题解]p-1= p1^a1 * p2^a2 * pk^ak,g是p的原根当且仅当对于所有的pi满足g^[ (p-1)/ ...

  2. img图片居中

    关键词:clear: both;    display: block;    margin:auto; 图片居左,居右,居中: /* Alignment */ .alignleft { display ...

  3. 使用generatorConfig工具自动生成mybatis的实体类以及dao接口和映射文件

    1:数据准备 创建一个数据库表 CREATE TABLE `logininfo` ( `id` ) NOT NULL AUTO_INCREMENT, `username` ) DEFAULT NULL ...

  4. koa源码阅读[3]-koa-send与它的衍生(static)

    koa源码阅读的第四篇,涉及到向接口请求方提供文件数据. 第一篇:koa源码阅读-0第二篇:koa源码阅读-1-koa与koa-compose第三篇:koa源码阅读-2-koa-router 处理静态 ...

  5. 环境变量配错了 command not found

    一般就是忘记在PATH 前面加$ 1.可以用whereis或者which命令查看一下有没有这个命令 具体执行which lswhereis ls 2.系统环境变量导致的问题解决方案: exportPA ...

  6. 目标检测-基于Pytorch实现Yolov3(1)- 搭建模型

    原文地址:https://www.cnblogs.com/jacklu/p/9853599.html 本人前段时间在T厂做了目标检测的项目,对一些目标检测框架也有了一定理解.其中Yolov3速度非常快 ...

  7. React 16 源码瞎几把解读 【三 点 一】 把react组件对象弄到dom中去(矛头指向fiber,fiber不解读这个过程也不知道)

    一.ReactDOM.render 都干啥了 我们在写react的时候,最后一步肯定是 ReactDOM.render( <div> <Home name="home&qu ...

  8. 91.Decode Ways---dp

    题目链接:https://leetcode.com/problems/decode-ways/description/ 题目大意:将给出的字符串解码,问有多少种解码方式.解码按照“ABC...Z&qu ...

  9. caffe Python API 之SoftmaxWithLoss

    net.loss = caffe.layers.SoftmaxWithLoss(net.fc3, net.label) 输出: layer { name: "loss" type: ...

  10. C++循环链表解决约瑟夫环问题

    约瑟夫环问题可以简单的使用数组的方式实现,但是现在我使用循环链表的方法来实现,因为上午看到一道面试题规定使用循环链表解决约瑟夫环问题. 什么是约瑟夫环? “约瑟夫环是一个数学的应用问题:已知n个人(以 ...