[HNOI2018]转盘
[HNOI2018]转盘
给你一个 \(n\) 元环, 你可以在 \(0\) 时刻从任意一个位置出发, 每一秒可以选择往后或者留在原地每个点有个参数 \(T_i\) , 当你走到 \(i\) 的时间 \(t\ge T_i\) 时你就可以把 \(i\) 标记问你把整个环上的点都标记最小需要多长时间, 带修改 \(T_i\), 强制在线.
\(\text{Solution:}\)
只看题目比较难发现其中的性质,我们来手模一下。
比如 \(T = 1, 2, 5, 4, 5\)
标记每个点的时间为 \(1, 2, 5, 6, 7\)
显然 \(3, 4, 5, 6, 7\) 也是一组合法解。
这启示我们如果从将题目转换一下,变成:
假设你 \(t\) 时刻在某个点,每次可以向前走或者留在原地,然后 \(t\) 减 \(1\),开始所有的点都出现,每个点在 \(T_i\) 时间消失,求一个最小的 \(t\) 使得在所有点都消失前访问所有点.
显然一直走会不会更差,如 \(3, 4, 5, 6, 7\) 不会比 \(1, 2, 5, 6, 7\) 差。
这下时间就是连续的,这也方便我们讨论。
再次简化:
在一个长度为 \(2n\) 的序列上(断环成链),从点 \(i\in[n,2n)\)出发
t-(i-j) \ge T_j\\
\Rightarrow t \ge T_j+i - j\\
\Rightarrow t_{min}= max\{ T_j-j\}+i
\]
求 \(t_{min}\)
仔细观察式子:

考虑枚举 \(j\) 看哪些 \(i\) 满足i的后缀最大值为 \(a_j\) ,从后往前扫,扫到一个数就更新后缀最大值,然后 \(chkmin\) 。
这样扫描每次 O(n),复杂度不对,没有充分利用每次修该前的信息,于是我们考虑将没有影响的一部分合并,可以用线段树来实现。
线段树每个节点维护该区间的最大 \(a\) ,以及答案,
向上合并的时候是在左区间递归找满足条件的 \(i\) ,往左还是往右找分类讨论一下,再选取最小的答案即可(比较难理解画一画,想一想)。
#include <set>
#include <cmath>
#include <cstdio>
#include <cstring>
#include <iostream>
#include <assert.h>
#include <algorithm>
using namespace std;
#define LL long long
#define debug(...) fprintf(stderr, __VA_ARGS__)
#define GO debug("GO\n")
inline int rint() {
register int x = 0, f = 1; register char c;
while (!isdigit(c = getchar())) if (c == '-') f = -1;
while (x = (x << 1) + (x << 3) + (c ^ 48), isdigit(c = getchar()));
return x * f;
}
template<typename T> inline void chkmin(T &a, T b) { a > b ? a = b : 0; }
template<typename T> inline void chkmax(T &a, T b) { a < b ? a = b : 0; }
const int maxN = 2e5 + 10;
int n, m, q;
int ans[maxN * 4], mx[maxN * 4], T[maxN];
#define ls (x<<1)
#define rs (x<<1|1)
int query(int x, int l, int r, int y) {
if (l == r) return l + max(mx[x], y);
int mid = (l + r) >> 1;
if (mx[rs] >= y) return min(ans[x], query(rs, mid + 1, r, y));
else return min(mid + 1 + y, query(ls, l, mid, y));
}
void pu(int x, int l, int r) {
mx[x] = max(mx[ls], mx[rs]);
ans[x] = query(ls, l, (l + r) >> 1, mx[rs]);
}
void Build(int x, int l, int r) {
if (l == r) {
mx[x] = ans[x] = T[l] - l;
return;
}
int mid = (l + r) >> 1;
Build(ls, l, mid);
Build(rs, mid + 1, r);
pu(x, l, r);
}
void change(int x, int l, int r, int y) {
if (l == r) {
ans[x] = mx[x] = T[l] - l;
return;
}
int mid = (l + r) >> 1;
if (y <= mid) change(ls, l, mid, y);
else change(rs, mid + 1, r, y);
pu(x, l, r);
}
int main() {
#ifndef ONLINE_JUDGE
freopen("xhc.in", "r", stdin);
freopen("xhc.out", "w", stdout);
#endif
n = rint(), m = rint(), q = rint();
for (int i = 1; i <= n; ++ i)
T[i] = T[i + n] = rint();
Build(1, 1, n * 2);
int lastans = ans[1] + n - 1;
printf("%d\n", lastans);
while (m --) {
int x = rint(), y = rint();
if (q) x ^= lastans, y ^= lastans;
T[x] = T[x + n] = y;
change(1, 1, 2 * n, x);
change(1, 1, 2 * n, x + n);
lastans = ans[1] + n - 1;
printf("%d\n", lastans);
}
}
[HNOI2018]转盘的更多相关文章
- 【BZOJ5286】[HNOI2018]转盘(线段树)
[BZOJ5286][HNOI2018]转盘(线段树) 题面 BZOJ 洛谷 题解 很妙的一道题目啊.(全世界除了我这题都有40分,就我是一个状压选手 首先来发现一些性质,我们走一圈一定不会更差. 为 ...
- 5286: [Hnoi2018]转盘
5286: [Hnoi2018]转盘 链接 分析: $\min\limits_{i=1}^n \{ \max\limits_{j=i}^{i + n - 1} \{ a_{j}+i \} \} +n- ...
- [BZOJ5286][洛谷P4425][HNOI2018]转盘(线段树)
5286: [Hnoi2018]转盘 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 15 Solved: 11[Submit][Status][Di ...
- bzoj 5286: [Hnoi2018]转盘
Description Solution 首先注意到一个点不会走两次,只会有停下来等待的情况,把序列倍长 那么如果枚举一个起点\(i\),答案就是 \(min(max(T[j]+n-(j-i)-1)) ...
- BZOJ.5286.[AHOI/HNOI2018]转盘(线段树)
BZOJ LOJ 洛谷 如果从\(1\)开始,把每个时间\(t_i\)减去\(i\),答案取决于\(\max\{t_i-i\}\).记取得最大值的位置是\(p\),答案是\(t_p+1+n-1-p=\ ...
- BZOJ5286: [Hnoi2018]转盘 (线段树)
题意 给你绕成一圈的物品共 \(n\) 个 , 然后从其中一个开始选 , 每次有两种操作 , 一是继续选择当前物品 , 二是选择这个后一个物品 . 选择后一个物品要求当前的时刻大于后一个的 \(T_i ...
- [HNOI2018]转盘[结论+线段树]
题意 题目链接 分析 首先要发现一个结论:最优决策一定存在一种 先在出发点停留之后走一圈 的情况,可以考虑如下证明: 如果要停留的话一定在出发点停留,这样后面的位置更容易取到. 走超过两圈的情况都可以 ...
- 【比赛】HNOI2018 转盘
通过这题,我发现了我最大的缺陷,就是题目中重要的性质发现不了,所以导致后期根本做不了.还是要多做题,培养思维 对于这道题,来发现性质吧 对于每一条路线,因为它有用的就是最终的时刻,所以我们都可以把它变 ...
- 【题解】HNOI2018转盘
何学长口中所说的‘一眼题’……然而实际上出出来我大HN全省也只有一个人A…… 首先我们需要发现一个性质:我们永远可以在最后一圈去标记所有的物品.倘若我们反复转圈,那么这完全是可以省下来的.所以我们破环 ...
随机推荐
- JAVA并发-线程状态
一.线程基本状态 新建:线程已创建但start()方法还没执行 就绪(可运行):start()方法已运行,但还没被选择 运行:从就绪线程中选择出某一个线程进行run()操作 阻塞(不可运行):线程正在 ...
- Web—02-轻松理解css
CSS基本语法以及页面引用 CSS基本语法 css的定义方法是: 选择器 { 属性:值; 属性:值; 属性:值;} 选择器是将样式和页面元素关联起来的名称,属性是希望设置的样式属性每个属性有一个或多个 ...
- Deferred Lighting
Deferred lighting separate lighting from rendering and make lighting a completely image-space techni ...
- iOS小知识点
1.子视图超出frame的部分不显示 view.clipsToBounds 设置为YES; UIScrollview的clipsToBounds默认就是YES 2.UIImage 有一个属性叫s ...
- 我的前端工具集(八)获得html元素在页面中的位置
我的前端工具集(八)获得html元素在页面中的位置 liuyuhang原创,未经允许禁止转载 目录 我的前端工具集 有时候需要用点击等操作,来获取某元素在页面中的位置,然后在该位置添加某些操作 如 ...
- 如何编写及运行JS
JS也是一种脚本语言,他可以有两种方式在HTML页面进行引入,一种是外联,一种是内部. 外联JS的写法为: <script src="相对路径"></ ...
- MySQL/MariaDB学习笔记——mysql.user表中存在多个root用户问题理解
mysql.user表中存在多个root用户问题 问题描述:使用 SELECT host,user FROM mysql.user 发现mysql.user表中存在三个root用户,如下 持着对中几个 ...
- 浅谈HashMap与线程安全 (JDK1.8)
HashMap是Java程序员使用频率最高的用于映射(键值对)处理的数据类型.HashMap 继承自 AbstractMap 是基于哈希表的 Map 接口的实现,以 Key-Value 的形式存在,即 ...
- Angular.js进阶
1.常用指令 <!DOCTYPE html> <html> <head lang="en"> <meta charset="UT ...
- Python文本操作2
# list3 = [# {"name": "alex", "hobby": "抽烟"},# {"name&q ...