题目链接

Description

给定一个长度为 \(N\) 的序列 \(a\),和一个区间 \([L, R]\)。

求多少连续子序列的权值和在区间内,即满足 \(1 \le i \le j \le n\) 且满足 \(L \le \sum_{k=i}^{j} a[i] \le R\) 的方案数。

Solution

区间和,很容易想到用前缀和转换,这样区间相关变成了两个点。设 \(s\) 为 \(a\) 的前缀和,那么统计就变成了这样。

统计 \(0 \le i < j \le n\) 中满足 \(L \le s[j] - s[i] \le R\) 的方案数的。数据只有一组询问,显然是支持我们枚举一维,的不妨枚举 \(s[i]\),那么转化一下式子,就是满足 \(L + s[i] \le s[j] \le R + s[i]\) 且 \(i < j\) 的 \(j\) 的数量。

这就是一个显然的二维偏序问题,做法就是:

  • 倒序枚举 \(i\)
  • 查询答案
  • 插入 \(s[i]\)

单调修改、区间查询这个操作我们再熟悉不过了。但是这次因为离散化会把值域信息搞没,所以不能离散化,只能动态开点线段树。(后来想了一下好像也可以,把数值全部打进数组离散化一下,所以写了两个版本)。

时间复杂度

\(O(n\log_2 10^{10})\)

Code

动态开点线段树版

#include <iostream>
#include <cstdio> using namespace std; typedef long long LL; const int N = 100005; int n, L, R, rt, idx; struct T{
int l, r, v;
} t[N * 30]; LL Lt = 9e18, Rt = -9e18; LL s[N], ans = 0; void inline pushup(int p) {
t[p].v = t[t[p].l].v + t[t[p].r].v;
} void insert(int &p, LL l, LL r, LL x) {
if (!p) p = ++idx;
t[p].v++;
if (l == r) return;
LL mid = (l + r) >> 1;
if (x <= mid) insert(t[p].l, l, mid, x);
else insert(t[p].r, mid + 1, r, x);
} int query(int p, LL l, LL r, LL x, LL y) {
if (!p) return 0;
if (x <= l && r <= y) return t[p].v;
LL mid = (l + r) >> 1, res = 0;
if (x <= mid) res += query(t[p].l, l, mid, x, y);
if (mid < y) res += query(t[p].r, mid + 1, r, x, y);
return res;
} int main() {
scanf("%d%d%d", &n, &L, &R);
for (int i = 1; i <= n; i++) scanf("%lld", &s[i]), s[i] += s[i - 1];
for (int i = 1; i <= n; i++) {
Lt = min(Lt, s[i]);
Rt = max(Rt, R + s[i]);
}
for (int i = n; ~i; i--) {
ans += query(rt, Lt, Rt, L + s[i], R + s[i]);
if (i) insert(rt, Lt, Rt, s[i]);
}
printf("%lld\n", ans);
return 0;
}

树状数组版

#include <iostream>
#include <cstdio>
#include <algorithm>
using namespace std; typedef long long LL; const int N = 100005; int n, L, R, tot, c[N]; LL s[N], d[N], ans = 0; int inline get(LL x) {
return lower_bound(d + 1, d + 1 + tot, x) - d;
} void inline add(int x) {
for (; x <= tot; x += x & -x) c[x]++;
} int inline ask(int x) {
int res = 0;
for (; x; x -= x & -x) res += c[x];
return res;
} int main() {
scanf("%d%d%d", &n, &L, &R);
for (int i = 1; i <= n; i++)
scanf("%lld", &s[i]), s[i] += s[i - 1], d[++tot] = s[i];
sort(d + 1, d + 1 + tot);
tot = unique(d + 1, d + 1 + tot) - d - 1;
for (int i = n; ~i; i--) {
int A = lower_bound(d + 1, d + 1 + tot, L + s[i]) - d - 1;
int B = upper_bound(d + 1, d + 1 + tot, R + s[i]) - d - 1;
ans += ask(B) - ask(A);
if (i) add(get(s[i]));
}
printf("%lld\n", ans);
return 0;
}

BJOI2016 回转寿司的更多相关文章

  1. P5459 [BJOI2016]回转寿司

    传送门 暴力怎么搞,维护前缀和 $s[i]$ ,对于每一个 $s[i]$,枚举所有 $j\in[0,i-1]$,看看 $s[i]-s[j]$ 是否属于 $[L,R]$ 如果属于就加入答案 $s[i]- ...

  2. BZOJ_4627_[BeiJing2016]回转寿司_离散化+树状数组

    BZOJ_4627_[BeiJing2016]回转寿司_离散化+树状数组 Description 酷爱日料的小Z经常光顾学校东门外的回转寿司店.在这里,一盘盘寿司通过传送带依次呈现在小Z眼前.不同的寿 ...

  3. bzoj 4627: [BeiJing2016]回转寿司 -- 权值线段树

    4627: [BeiJing2016]回转寿司 Time Limit: 10 Sec  Memory Limit: 256 MB Description 酷爱日料的小Z经常光顾学校东门外的回转寿司店. ...

  4. bzoj 4627: [BeiJing2016]回转寿司

    4627: [BeiJing2016]回转寿司 Description 酷爱日料的小Z经常光顾学校东门外的回转寿司店.在这里,一盘盘寿司通过传送带依次呈现在小Z眼前.不同的寿 司带给小Z的味觉感受是不 ...

  5. 【BZOJ4627】[BeiJing2016]回转寿司 SBT

    [BZOJ4627][BeiJing2016]回转寿司 Description 酷爱日料的小Z经常光顾学校东门外的回转寿司店.在这里,一盘盘寿司通过传送带依次呈现在小Z眼前.不同的寿司带给小Z的味觉感 ...

  6. [LOJ2736] [JOISC 2016 Day 3] 回转寿司 (分块+堆)

    [LOJ2736] [JOISC 2016 Day 3] 回转寿司 (分块+堆) 题面 给出一个有n 个点的环,环上各点有一个初始权值 \(a_i\) 给出 Q 个询问,每次询问给出一个区间 [l,r ...

  7. K - 回转寿司(值域段数(板题) + 动态开点)

    回转寿司 Description 酷爱日料的小Z经常光顾学校东门外的回转寿司店.在这里,一盘盘寿司通过传送带依次呈现在小Z眼前.不同的寿 司带给小Z的味觉感受是不一样的,我们定义小Z对每盘寿司都有一个 ...

  8. 【bzoj4627】[BeiJing2016]回转寿司 离散化+树状数组

    题目描述 给出一个长度为n的序列,求所有元素的和在[L,R]范围内的连续子序列的个数. 输入 第一行包含三个整数N,L和R,分别表示寿司盘数,满意度的下限和上限. 第二行包含N个整数Ai,表示小Z对寿 ...

  9. bzoj4627: [BeiJing2016]回转寿司

    权值线段树. 要求 L<=(s[i]-s[j])<=R  (i<j). 的i和j的数量. 所以把前缀和s加入一棵权值线段树,每次询问满足条件的范围中的权值的个数. 权值线段树不能像普 ...

随机推荐

  1. 'sortbitwise'是什么意思

    问题 flag sortbitwise 在ceph中是什么意思,在Jewel版本下可以看到多了这个flags [root@lab8106 current]# ceph -s cluster ffe7a ...

  2. 常用linux源列表

    本篇记录一些常用的源文件,后面需要用到的时候,直接进行复制粘贴即可 centos 相关 base源 [base] name=CentOS-$releasever - Base - mirrors.al ...

  3. mysql学习——数据表基本操作1

    选择数据库 创建数据表 包括字段名和数据类型两部分 查看数据表 使用主键约束 主键又称主码,能够唯一的表示表中的一条记录,分为单字段主键与多字段联合主键 单字段主键 定义主键的两种方式: 1.在最后指 ...

  4. 在家看电影音效太差?Boom 3D帮你轻松升级

    新片上映后,很多人都会选择去电影院观看,一是为了第一时间看到电影,还有一个原因就是电影院的音效往往可以让人身临其境,更好地感受电影的氛围.那如果在家刷片我们该怎么办呢? 我们可以使用Boom 3D这款 ...

  5. 工作中使用mongodb

    写了一个mongodb的基类 1 <?php 2 3 namespace BI\Service\MongoDB; 4 5 use MongoDB\Driver\BulkWrite; 6 use ...

  6. HarmonyOS Java UI之DependentLayout布局示例

    DependentLayout简介 DependentLayout意为相对位置布局,与DirectionalLayout相比较有更多的排布方式,每个组件可以指定相对于其他同级组件的位置,也可以指定相对 ...

  7. @Transactional自调用问题

  8. 现代富文本编辑器Quill的模块化机制

    DevUI是一支兼具设计视角和工程视角的团队,服务于华为云DevCloud平台和华为内部数个中后台系统,服务于设计师和前端工程师.官方网站:devui.designNg组件库:ng-devui(欢迎S ...

  9. java中的强引用(Strong reference),软引用(SoftReference),弱引用(WeakReference),虚引用(PhantomReference)

    之前在看深入理解Java虚拟机一书中第一次接触相关名词,但是并不理解,只知道Object obj = new Object()类似这种操作的时候,obj就是强引用.强引用不会被gc回收直到gc roo ...

  10. centOs7.5.64之前的操作系统搭建GitLab记录

    GitLab搭建步骤: 1. Install and configure the necessary dependencies (1)yum install curl openssh-server o ...