GMOJ5673 爬山法 题解
Solution
显然先想到处理出每个点能看到的最高的顶点。
然后考虑模拟题目的过程,一段一段走时间复杂度显然不够优秀。
考虑我们要求什么,我们需要求出\(u\)到\(v\)的最近的一个点,使得这个点能看到的点比\(v\)能看到的点更高。
然后这个东西可以直接线段树,当然也可以二分+st表
复杂度\(O(n\log n)\)
Code
#include <cstdio>
#include <iostream>
#define LL long long
#define RE register
#define IN inline
using namespace std;
IN int read() {
int res = 0; char ch = getchar();
for(; !isdigit(ch); ch = getchar());
for(; isdigit(ch); ch = getchar()) res = (res << 1) + (res << 3) + (ch ^ 48);
return res;
}
int n, le[2000010], ri[2000010], nxt[2000010], st[2000010][20], lg[2000010];
LL f[2000010];
struct Point {
int x, y;
}p[2000010];
inline double slope(int x, int y) {return 1.0 * (p[x].y - p[y].y) / (p[x].x - p[y].x);}
inline bool cmp(int x, int y) {
if(x == y) return false;
if(p[x].y > p[y].y) return true;
if(p[x].y == p[y].y) return p[x].x > p[y].x;
return false;
}
int stk[2000010];
inline int pmax(int x, int y) {return cmp(x, y) ? x : y;}
void preWork() {
int top = 0;
for(int i = 1; i <= n; ++i) {
while(top > 1 && slope(stk[top], i) >= slope(stk[top], stk[top - 1])) -- top;
if(top) le[i] = stk[top]; stk[++top] = i;
}
top = 0;
for(int i = n; i; --i) {
while(top > 1 && slope(stk[top], i) <= slope(stk[top], stk[top - 1])) -- top;
if(top) ri[i] = stk[top]; stk[++top] = i;
}
for(int i = 1; i <= n; ++i) {
nxt[i] = pmax(le[i], ri[i]);
if(cmp(i, nxt[i])) nxt[i] = 0;
st[i][0] = nxt[i];
}
for(int i = 1; i <= lg[n]; ++i)
for(int j = 1; j <= n; ++j)
st[j][i] = pmax(st[j][i - 1], st[j + (1 << i - 1)][i - 1]);
return ;
}
inline LL abs(int x) {return x < 0 ? -x : x;}
inline int query(int l, int r) {int len = r - l + 1; return pmax(st[l][lg[len]], st[r - (1 << lg[len]) + 1][lg[len]]);}
LL solve(int k) {
if(f[k] || !nxt[k]) return f[k];
int res = nxt[k];
if(nxt[k] < k) {
for(int l = nxt[k], r = k - 1, mid; l <= r;) {
mid = l + r >> 1;
if(cmp(query(mid, k), nxt[k])) res = mid, l = mid + 1;
else r = mid - 1;
}
}
else {
for(int l = k + 1, r = nxt[k], mid; l <= r;) {
mid = l + r >> 1;
if(cmp(query(k, mid), nxt[k])) res = mid, r = mid - 1;
else l = mid + 1;
}
}
f[k] = solve(res) + abs(res - k);
return f[k];
}
void out(int x) {printf("%d %d\n",p[x].x,p[x].y);}
int main() {
freopen("mountain.in","r",stdin);
freopen("mountain.out","w",stdout);
n = read();
for(int i = 2; i <= n; ++i) lg[i] = lg[i >> 1] + 1;
for(int i = 1; i <= n; ++i) p[i].x = read(), p[i].y = read();
preWork();
for(int i = 1; i <= n; ++i) solve(i);
for(int i = 1; i <= n; ++i) printf("%lld\n",f[i]);
return 0;
}```
GMOJ5673 爬山法 题解的更多相关文章
- 2016 华南师大ACM校赛 SCNUCPC 非官方题解
我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...
- noip2016十连测题解
以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...
- BZOJ-2561-最小生成树 题解(最小割)
2561: 最小生成树(题解) Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1628 Solved: 786 传送门:http://www.lyd ...
- Codeforces Round #353 (Div. 2) ABCDE 题解 python
Problems # Name A Infinite Sequence standard input/output 1 s, 256 MB x3509 B Restoring P ...
- 哈尔滨理工大学ACM全国邀请赛(网络同步赛)题解
题目链接 提交连接:http://acm-software.hrbust.edu.cn/problemset.php?page=5 1470-1482 只做出来四道比较水的题目,还需要加强中等题的训练 ...
- 2016ACM青岛区域赛题解
A.Relic Discovery_hdu5982 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Jav ...
- poj1399 hoj1037 Direct Visibility 题解 (宽搜)
http://poj.org/problem?id=1399 http://acm.hit.edu.cn/hoj/problem/view?id=1037 题意: 在一个最多200*200的minec ...
- 网络流n题 题解
学会了网络流,就经常闲的没事儿刷网络流--于是乎来一发题解. 1. COGS2093 花园的守护之神 题意:给定一个带权无向图,问至少删除多少条边才能使得s-t最短路的长度变长. 用Dijkstra或 ...
- CF100965C题解..
求方程 \[ \begin{array}\\ \sum_{i=1}^n x_i & \equiv & a_1 \pmod{p} \\ \sum_{i=1}^n x_i^2 & ...
随机推荐
- NFS 服务器配置(Ubuntu)
# NFS 服务器配置(Ubuntu 20.0) # 1.配置网络环境 # NFS 的客户端和服务端必须在同一局域网 # 2.在服务器上安装nfs sudo apt-get install nfs-c ...
- Redis 08 地理位置
参考源 https://www.bilibili.com/video/BV1S54y1R7SB?spm_id_from=333.999.0.0 版本 本文章基于 Redis 6.2.6 Redis 的 ...
- java学习第一天.day02
整数类型常量 整数类型的常量JVM默认使用 int 类型来存储 小数类型类型 小数类型的常量JVM默认使用 double 类型来存储 . ASCII表 A在码表的顺序是65,a在码表的顺序是97
- PerfView专题 (第八篇):洞察 C# 内存泄漏之寻找静态变量名和GC模式
一:背景 这篇我们来聊一下 PerfView 在协助 WinDbg 分析 Dump 过程中的两个超实用技巧,可能会帮助我们快速定位最后的问题,主要有如下两块: 洞察内存泄漏中的静态大集合变量名. 验证 ...
- GIL互斥锁与线程
GIL互斥锁与线程 GIL互斥锁验证是否存在 """ 昨天我们买票的程序发现很多个线程可能会取到同一个值进行剪除,证明了数据是并发的,但是我们为了证明在Cpython中证 ...
- Star (欧拉函数)
题面 Fernando won a compass for his birthday, and now his favorite hobby is drawing stars: first, he ma ...
- JOIOI王国 - 二分+贪心
题面 题解 通过一句经典的话"最大值的最小值" 我判断它是二分题, 不难发现,整个图形中两个省的分界线是一条单调不递减或单调不递增的折线. 而且,越到后来它的最大值只会越来越大,最 ...
- 长篇图解etcd核心应用场景及编码实战
大家好啊,我是字母哥,今天写一篇关于etcd的文章,其实网上也有很多关于etcd的介绍,我就简明扼要,总结提炼,期望大家通过这一篇文章掌握etcd的核心知识以及编码技能! 本文首先用大白话给大家介绍一 ...
- Python之创建数据库及功能示例样本
创建数据库实例 import pymysql db= pymysql.connect(host="localhost",user="root",password ...
- 简析XDP的重定向机制
GreatSQL社区原创内容未经授权不得随意使用,转载请联系小编并注明来源. GreatSQL是MySQL的国产分支版本,使用上与MySQL一致. 一. XDP Socket示例解析 源码参见:htt ...