BZOJ 5308 [ZJOI2018] Day2T2 胖 | 二分 ST表
题目链接
题解
这么简单的题
为什么考场上我完全想不清楚 = =
对于k个关键点中的每一个关键点\(a\),二分它能一度成为哪些点的最短路起点(显然这些点在一段包含\(a\)的连续区间中),所以二分这个区间的左右端点。
如何判断某个点\(p\)是否在这个区间内呢?设\(d = |a - p|\),则\(a\)可以更新\(p\)当且仅当区间\([p - d, p + d]\)中的关键点到\(p\)的距离没有比\(a\)更优的。
设\(sum_i\)点\(i\)到点\(1\)的距离,用ST表维护一个关键点区间中\((sum_i - l)_{\max}\)和\((sum_i + l)_{\min}\),则\([p - d, p + d]\)中对于\(p\)的最优关键点可能是\([p - d, p]\)中\(sum_i - l\)最大的或\((p, p+d]\)中\(sum_i + l\)最小的。将他俩与\(a\)比较,如果\(a\)仍然是最优的,则\(a\)可以一度占领\(p\)这个点。
注意若两个关键点到\(p\)的距离相同,只能算一个,那么人为规定一下【离\(p\)边数最小的】是最优的,如果两个点这一项也一样,则取编号较小的即可。
#include <cstdio>
#include <cstring>
#include <cmath>
#include <algorithm>
#include <iostream>
#define enter putchar('\n')
#define space putchar(' ')
using namespace std;
typedef long long ll;
template <class T>
void read(T &x){
char c;
bool op = 0;
while(c = getchar(), c < '0' || c > '9')
if(c == '-') op = 1;
x = c - '0';
while(c = getchar(), c >= '0' && c <= '9')
x = x * 10 + c - '0';
if(op == 1) x = -x;
}
template <class T>
void write(T x){
if(x < 0) putchar('-'), x = -x;
if(x >= 10) write(x / 10);
putchar('0' + x % 10);
}
const int N = 200005;
const ll INF = 1e18 + 233;
int n, m, K, lg[N];
ll sum[N], st1[N][20], st2[N][20], l1[N], l2[N];
struct Edge {
int a;
ll l;
Edge(){}
Edge(int x, ll y): a(x), l(y){}
bool operator < (const Edge &b) const {
return a < b.a;
}
} e[N];
int Max(int a, int b){
if(l1[a] == l1[b]) return max(a, b);
return l1[a] > l1[b] ? a : b;
}
int Min(int a, int b){
if(l2[a] == l2[b]) return min(a, b);
return l2[a] < l2[b] ? a : b;
}
void init(){
sort(e + 1, e + K + 1);
for(int i = 1; i <= K; i++){
l1[i] = sum[e[i].a] - e[i].l, l2[i] = sum[e[i].a] + e[i].l;
st1[i][0] = st2[i][0] = i;
}
for(int j = 1; (1 << j) <= K; j++)
for(int i = 1; i + (1 << j) - 1 <= K; i++){
st1[i][j] = Max(st1[i][j - 1], st1[i + (1 << (j - 1))][j - 1]);
st2[i][j] = Min(st2[i][j - 1], st2[i + (1 << (j - 1))][j - 1]);
}
}
int get1(int l, int r){
int j = lg[r - l + 1];
return Max(st1[l][j], st1[r - (1 << j) + 1][j]);
}
int get2(int l, int r){
int j = lg[r - l + 1];
return Min(st2[l][j], st2[r - (1 << j) + 1][j]);
}
int better(int a, int b, int p){
ll ans1 = e[a].l + abs(sum[p] - sum[e[a].a]);
ll ans2 = e[b].l + abs(sum[p] - sum[e[b].a]);
if(ans1 != ans2) return ans1 < ans2 ? a : b;
if(abs(e[a].a - p) != abs(e[b].a - p))
return abs(e[a].a - p) < abs(e[b].a - p) ? a : b;
return min(a, b);
}
bool bel(int p, int k){
int d = abs(e[k].a - p), best = k;
int l = lower_bound(e + 1, e + K + 1, Edge(p - d, 0)) - e;
int r = upper_bound(e + 1, e + K + 1, Edge(p + d, 0)) - e - 1;
int mid = upper_bound(e + 1, e + K + 1, Edge(p, 0)) - e - 1;
if(l <= mid) best = better(best, get1(l, mid), p);
if(mid < r) best = better(best, get2(mid + 1, r), p);
return best == k;
}
ll solve(){
ll ans = 0;
for(int i = 1; i <= K; i++){
int l = 1, r = e[i].a, mid, L;
while(l < r)
if(bel(mid = (l + r) >> 1, i)) r = mid;
else l = mid + 1;
L = l, l = e[i].a, r = n;
while(l < r)
if(bel(mid = (l + r + 1) >> 1, i)) l = mid;
else r = mid - 1;
ans += (r - L + 1);
}
return ans;
}
int main(){
for(int i = 2; i < N; i++)
lg[i] = lg[i >> 1] + 1;
read(n), read(m);
for(int i = 2; i <= n; i++)
read(sum[i]), sum[i] += sum[i - 1];
while(m--){
read(K);
for(int i = 1; i <= K; i++)
read(e[i].a), read(e[i].l);
init();
write(solve()), enter;
}
return 0;
}
BZOJ 5308 [ZJOI2018] Day2T2 胖 | 二分 ST表的更多相关文章
- 「ZJOI2018」胖(ST表+二分)
「ZJOI2018」胖(ST表+二分) 不开 \(O_2\) 又没卡过去是种怎么体验... 这可能是 \(ZJOI2018\) 最简单的一题了...我都能 \(A\)... 首先我们发现这个奇怪的图每 ...
- ZJOI2018 胖 二分 ST表
原文链接https://www.cnblogs.com/zhouzhendong/p/ZJOI2018Day2T2.html 题目传送门 - BZOJ5308 题目传送门 - LOJ2529 题目传送 ...
- BZOJ4556:[TJOI\HEOI2016]字符串(后缀数组,主席树,二分,ST表)
Description 佳媛姐姐过生日的时候,她的小伙伴从某东上买了一个生日礼物.生日礼物放在一个神奇的箱子中.箱子外边写了一个长为n的字符串s,和m个问题.佳媛姐姐必须正确回答这m个问题,才能打开箱 ...
- BZOJ3166 [Heoi2013]Alo 【可持久化trie树 + 二分 + ST表】
题目 Welcome to ALO ( Arithmetic and Logistic Online).这是一个VR MMORPG , 如名字所见,到处充满了数学的谜题. 现在你拥有n颗宝石,每颗宝石 ...
- [BZOJ4310] 跳蚤 - 后缀数组,二分,ST表
[BZOJ4310] 跳蚤 Description 首先,他会把串分成不超过 \(k\) 个子串,然后对于每个子串 \(S\) ,他会从 \(S\) 的所有子串中选择字典序最大的那一个,并在选出来的 ...
- 洛谷P4501/loj#2529 [ZJOI2018]胖(ST表+二分)
题面 传送门(loj) 传送门(洛谷) 题解 我们对于每一个与宫殿相连的点,分别计算它会作为多少个点的最短路的起点 若该点为\(u\),对于某个点\(p\)来说,如果\(d=|p-u|\),且在\([ ...
- BZOJ 3230 相似子串 | 后缀数组 二分 ST表
BZOJ 3230 相似子串 题面 题解 首先我们要知道询问的两个子串的位置. 先正常跑一遍后缀数组并求出height数组. 对于每一个后缀suffix(i),考虑以i开头的子串有多少是之前没有出现过 ...
- bzoj 3230: 相似子串【SA+st表+二分】
总是犯低级错误,st表都能写错-- 正反分别做一遍SA,预处理st表方便查询lcp,然后处理a[i]表示前i个后缀一共有多少个本质不同的子串,这里的子串是按字典序的,所以询问的时候直接在a上二分排名就 ...
- HDU5726 GCD(二分 + ST表)
题目 Source http://acm.hdu.edu.cn/showproblem.php?pid=5726 Description Give you a sequence of N(N≤100, ...
随机推荐
- DeskMini无传统机械键盘与鼠标接口的情况下使用U盘安装系统经验总结
总结安装纯净版Win7旗舰版系统安装过程所解决的问题要点: 1:UEFI引导启动的实现. 2:使用Dism++实现系统的安装. 3:使用Dism++解决新主板在安装系统过程中不能使用USB键盘和鼠标的 ...
- 【UFUN开发板评测】小巧而不失精致,简单而不失内涵——uFun开发板开箱爆照
关于uFun学习板--"满满的爱和正能量" uFun是由@张进东 张工组织发起的一个开源的学习板,设计初衷是为了帮助学生更好的理解电子知识和开发技巧,同时又能对学生毕业找工作有很明 ...
- 编写脚本自动部署反向代理、web、nfs
服务器端 #!/bin/bash function nginx_install(){ if [[ -f /usr/sbin/nginx ]]; then echo 'Nginx has been in ...
- 针对Nginx日志的相关运维操作记录
在分析服务器运行情况和业务数据时,nginx日志是非常可靠的数据来源,而掌握常用的nginx日志分析命令的应用技巧则有着事半功倍的作用,可以快速进行定位和统计. 1)Nginx日志的标准格式(可参考: ...
- Pupet自动化管理环境部署记录
废话不多说了,下面记录下Puppet在Centos下的部署过程: puppet是什么puppet是一种基于ruby语言开发的Lnux.Unix.windows平台的集中配置管理系统.它使用自有的pup ...
- 安卓开发helloworld
https://blog.csdn.net/tangjie134/article/details/79495204
- TestSushu1
https://github.com/jzjaerui/Individual-Project/blob/master/TestSushu1 <程序设计实践I> 题目: ...
- 剑值offer:最小的k个数
这是在面试常遇到的topk问题. 题目描述: 输入n个整数,找出其中最小的K个数.例如输入4,5,1,6,2,7,3,8这8个数字,则最小的4个数字是1,2,3,4,. 解题思路: 思路一:用快排对数 ...
- NODE中解决跨域请求的问题
1.Node Express 解决请求跨域请求 标签(空格分隔): 跨域 1是Access-Control-Allow-Origin 允许的域 2是Access-Control-Allow-Heade ...
- DockerHub使用简介
常用的Docker镜像文件都有,就不用自己费劲的一点点配置了,这才是Docker的真正目的.就像Ghost里边含office,直接还原,不用一台台机器安装呢,省时省力,与高效工作的理念相契合. 至于, ...