[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…… 首先我们需要发现一个性质:我们永远可以在最后一圈去标记所有的物品.倘若我们反复转圈,那么这完全是可以省下来的.所以我们破环 ...
随机推荐
- sizeof笔试题--转
转自http://blog.csdn.net/yanyaohua0314/archive/2007/09/17/1787749.aspx sizeof笔试题 http://www.xici.net/b ...
- HttpClient请求地址并携带参数
废话不多,直接上代码! /** * * @param httpUrl * @param reqInfo * @return */ public static StringBuffer sendPost ...
- NDK-C++ support
1.NDK相关各种可用的C++运行库Android平台自带微型C++运行库(system),NDK提供补充功能的C++运行库(gabi++, stlport, gnustl)运行库 异常支持 RTTI ...
- (二、下) springBoot 、maven 、mysql、 mybatis、 通用Mapper、lombok 简单搭建例子 《附项目源码》
接着上篇文章中 继续前进. 一.在maven 的pom.xm中添加组件依赖, mybatis通用Mapper,及分页插件 1.mybatis通用Mapper <!-- mybatis通用Mapp ...
- CoacoaPods安装使与使用超级详细教程
对于一个iOS开发的初学者来说,并不知道第三方类库的存在,知道了也不知道如何使用,那么下面便来介绍一下使用方法. iOS开发常用的第三方类库是GitHub:https://github.com/ 在上 ...
- oracle入门(一)
### 一.体系结构 1. 数据库 : 只有一个数据库 2. 实例 : 后台运行的一个进程 3. 表空间: 逻辑存储单位 4. 数据文件: 物理存储单位 5. 用户:面向用户管理,由用户来管理表空间, ...
- springboot-redis缓存
Redis缓存使用 1. 引入依赖(可能已经引入了):spring-boot-starter-cache 2. 在application.yml配置文件中配置spring:redis:host/p ...
- mysql中用HEX和UNHEX函数处理二进制数据的导入导出
读取数据并拼写sql语句,然后进行导入.具体方法为: (1)导出时采用HEX函数读取数据,把二进制的数据转为16进制的字符串: select HEX(binField) from testTable; ...
- VMware10安装CentOS7
先去网上下载一个VMware的破解版或者激活版,安装配置这里就不介绍了自行下载安装,基本过程相当于windows下安装个软件而已. CentOS7镜像下载就下阿里云站点的这是链接 http://mir ...
- Mysql是否开启binlog日志&开启方法
运行sql show variables like 'log_bin'; 如果Value 为 OFF 则为开启日志文件 如何开启mysql日志? 找到my,cnf 中 [mysqld] 添加如下 ...