洛谷 P3644 [APIO2015]八邻旁之桥(对顶堆维护中位数)
题意:
一条河将大地分为 \(A,B\) 两个部分。两部分均可视为一根数轴。
有 \(n\) 名工人,第 \(i\) 名的家在 \(x_i\) 区域的 \(a_i\) 位置,公司在 \(y_i\) 区域的 \(b_i\) 位置。
现在你可以建立 \(k\) 座桥,在 \(x\) 位置建立一座桥可以连接 \(A\) 区域的 \(x\) 位置和 \(B\) 区域的 \(x\) 位置,桥长为 \(1\) 个单位长度。
设 \(d_i\) 为第 \(i\) 名工人从家到公司走过的最短距离,求 \(D=d_1+d_2+\dots+d_n\) 的最小值。
\(n \in [1,2\times 10^5],k\in\{1,2\},a_i,b_i \leq 10^9,x_i,y_i \in \{'A','B'\}\)。
一开始看错题了,以为 \(k\) 的数据范围也是 \(10^5\)。。。。。。然后就不愿意继续想下去了。
首先如果家和公司在河同一边那方案肯定是唯一的,直接加上 \(|a_i-b_i|\)。
接下来重点考虑家和公司不在河同一边的情况,假设这些工人 \(d_i\) 的和为 \(D'\)
先从 \(k=1\) 入手,假设我们在 \(p\) 位置建了座桥,那么所有家和公司不在河同一边都必须通过这一座桥,即 \(d_i=|a_i-p|+|b_i-p|+1\)。
\(D'=\sum |a_i-p|+|b_i-p|+1\)
最后那个 \(1\) 显然可以直接处理掉,剩余部分就是一个初一弱智数学问题,直接取中位数所有 \(a_i,b_i\) 就可以了。
接下来考虑 \(k=2\) 的情况。假设我们在 \(p,q\) 位置建了桥。
那么 \(d_i\) 就是从这两座桥上通过所需的距离的较小值,即 \(d_i=\min(|a_i-p|+|b_i-p|,|a_i-q|+|b_i-q|)+1\)。
在 \(k=1\) 的情况中,我们之所以能够把 \(a_i,b_i\) 揉在一起取中位数,是因为它们的贡献互不影响。
但在这种情况下就不能直接取中位数了,因为有个 \(\min\)。
不妨设 \(a_i \leq b_i\),设 \(f(x)=|a_i-x|+|b_i-x|\)。简单画个图像,由三部分组成,左边是斜率为 \(-2\) 的射线,中间是一段水平线段,右边是斜率为 \(2\) 的射线。图像的对称轴为 \(x=\dfrac{a_i+b_i}{2}\)(梦回课内)
由于对称轴左右两边完全一样并且对称轴左边 \(y\) 随 \(x\) 的增大单调不降,故有:
- 若 \(|\dfrac{a_i+b_i}{2}-p|<|\dfrac{a_i+b_i}{2}-q|\),那么 \(f(p)\leq f(q)\)。
有了这个结论,本题就简单多了。
把所有工人按 \(\dfrac{a_i+b_i}{2}\) 从小到大排序,那我们肯定是选择离 \(\dfrac{a_i+b_i}{2}\) 较近的那座桥。
不妨设 \(p<q\),那么所有 \(\dfrac{a_i+b_i}{2}\leq\dfrac{p+q}{2}\) 的工人都会选择 \(p\) 那座桥,剩余的工人会选择 \(q\) 那座桥。
故选择 \(p\) 的工人是原序列的一个前缀,选择 \(q\) 的工人是原序列的一个后缀。
枚举断点 \(i\),\([1,i]\) 的工人选择桥梁 \(p\),\([i+1,n]\) 的工人选择桥梁 \(q\)。
断点两边是互相独立的。这时候我们又可以把绝对值拆开,于是我们又回到了第一问。
于是题目变为如何快速求出每个前缀的中位数,有 DS 味儿了,稍微一想就可以想到平衡树(
但稍微想一想就发现,其实根本不用平衡树。可以用两个堆来维护,建一个大根堆维护前一半的值,再建一个小根堆维护后一半的值。每次新插入一个值,就将它插入到对应的部分中,然后通过微调使两部分平衡。
#include <bits/stdc++.h>
using namespace std;
#define fi first
#define se second
#define fz(i,a,b) for(int i=a;i<=b;i++)
#define fd(i,a,b) for(int i=a;i>=b;i--)
#define ffe(it,v) for(__typeof(v.begin()) it=v.begin();it!=v.end();it++)
#define fill0(a) memset(a,0,sizeof(a))
#define fill1(a) memset(a,-1,sizeof(a)
#define fillbig(a) memset(a,63,sizeof(a))
#define pb push_back
#define ppb pop_back
#define mp make_pair
typedef pair<int,int> pii;
typedef long long ll;
const int MAXN=1e5+5;
struct solver{
multiset<int> st1,st2;
ll sum1=0,sum2=0;
void insert(int a){
if(st1.empty()){
st1.insert(a);sum1+=a;
} else {
int x=*st1.rbegin();
if(a<=x) sum1+=a,st1.insert(a);
else sum2+=a,st2.insert(a);
int cnt=(st1.size()+st2.size()+1)/2;
while(st1.size()>cnt){int v=*st1.rbegin();sum1-=v;sum2+=v;st1.erase(st1.find(v));st2.insert(v);}
while(st1.size()<cnt){int v=*st2.begin();sum2-=v;sum1+=v;st2.erase(st2.find(v));st1.insert(v);}
}
}
ll query(){
if(st1.empty()) return 0;
int cnt=(st1.size()+st2.size()+1)/2,x=*st1.rbegin();
return 1ll*cnt*x-sum1+sum2-1ll*(st1.size()+st2.size()-cnt)*x;
}
} x1,x2;
int n,k,m;
struct data{
int a,b;
friend bool operator <(data x,data y){
return x.a+x.b<y.a+y.b;
}
} f[MAXN];
ll sum=0;
ll pre[MAXN],suf[MAXN];
int main(){
scanf("%d%d",&k,&n);
for(int i=1;i<=n;i++){
char x,y;int a,b;cin>>x>>a>>y>>b;
if(x==y) sum+=abs(a-b);
else f[++m].a=a,f[m].b=b,sum++;
}
if(k==1){
for(int i=1;i<=m;i++) x1.insert(f[i].a),x1.insert(f[i].b);
printf("%lld\n",x1.query()+sum);
} else {
sort(f+1,f+m+1);
for(int i=1;i<=m;i++) x1.insert(f[i].a),x1.insert(f[i].b),pre[i]=x1.query();
for(int i=m;i;i--) x2.insert(f[i].a),x2.insert(f[i].b),suf[i]=x2.query();
ll mn=1e18;
for(int i=1;i<=m+1;i++) mn=min(mn,pre[i-1]+suf[i]);
printf("%lld\n",mn+sum);
}
return 0;
}
洛谷 P3644 [APIO2015]八邻旁之桥(对顶堆维护中位数)的更多相关文章
- 洛谷 P3644 [APIO2015]八邻旁之桥 解题报告
P3644 [APIO2015]八邻旁之桥 题目描述 一条东西走向的穆西河将巴邻旁市一分为二,分割成了区域\(A\)和区域\(B\). 每一块区域沿着河岸都建了恰好\(1000000001\)栋的建筑 ...
- [APIO2015]八邻旁之桥——非旋转treap
题目链接: [APIO2015]八邻旁之桥 对于$k=1$的情况: 对于起点和终点在同侧的直接计入答案:对于不在同侧的,可以发现答案就是所有点坐标与桥坐标的差之和+起点与终点不在同一侧的人数. 将所有 ...
- [BZOJ4071][APIO2015]八邻旁之桥
BZOJ(这题是BZOJ权限题,有权限号的就去看看吧) Luogu(良心洛谷) 题目描述 一条东西走向的穆西河将巴邻旁市一分为二,分割成了区域\(A\)和区域\(B\). 每一块区域沿着河岸都建了恰好 ...
- [APIO2015]八邻旁之桥
题面在这里 sol 这是一个\(Splay\)的题解 首先,如果一个人的家和办公室在同一侧,我们可以直接预处理; 如果不在同一侧,也可以加上1(当然要过桥啦) 当k==1时 我们设桥的位置为\(pos ...
- 题解【luoguP3644 [APIO2015]八邻旁之桥】
题目链接 题解 家和公司在同侧 简单,直接预处理掉 若 \(k=1\) 取所有的居民的\(\frac{家坐标+公司坐标}{2}\)的所有坐标的正中间建一座桥,使所有居民到的距离最小. 实现方法:线段树 ...
- [luoguP3644] [APIO2015]八邻旁之桥(权值线段树)
传送门 首先如果起点终点都在同一侧可以直接处理,如果需要过桥答案再加1 对于k等于1的情况 桥的坐标为x的话,a和b为起点和终点坐标 $ans=\sum_{1}^{n} abs(a_{i}-x)+ab ...
- APIO2015 八邻旁之桥/巴邻旁之桥
题目描述: bz luogu 题解: 贪心+权值线段树. $K=1$的时候,答案为$\sum |x-l| + |x-r|$,所以所有端点排序后取中位数即可. $K=2$的时候,一定是左边的一些走左边的 ...
- 【BZOJ4071】八邻旁之桥(线段树)
[BZOJ4071]八邻旁之桥(线段树) 题面 BZOJ权限题,洛谷链接 题解 既然\(k<=2\) 那么,突破口就在这里 分类讨论 ①\(k=1\) 这...不就是中位数吗.... 直接把所有 ...
- 【BZOJ4071】[Apio2015]巴邻旁之桥 Treap
[BZOJ4071][Apio2015]巴邻旁之桥 Description 一条东西走向的穆西河将巴邻旁市一分为二,分割成了区域 A 和区域 B. 每一块区域沿着河岸都建了恰好 1000000001 ...
随机推荐
- 更好的 java 重试框架 sisyphus 入门简介
What is Sisyphus sisyphus 综合了 spring-retry 和 gauva-retrying 的优势,使用起来也非常灵活. 为什么选择这个名字 我觉得重试做的事情和西西弗斯很 ...
- 打造专属测试平台4-使用Docker部署Django项目
编写完项目代码后,为了稳定的运行,需要将其部署至服务器.这里我选择了Docker去部署Django后端代码. 首先来看看Runoob对Docker的介绍: Docker 是一个开源的应用容器引擎,基于 ...
- 论文解读丨表格识别模型TableMaster
摘要:在此解决方案中把表格识别分成了四个部分:表格结构序列识别.文字检测.文字识别.单元格和文字框对齐.其中表格结构序列识别用到的模型是基于Master修改的,文字检测模型用到的是PSENet,文字识 ...
- [对对子队]测试报告Beta
一.测试中发现的bug BETA阶段的新bug 描述 提出者(可能需要发现者在会议上复现) 处理人 是否解决 第四关中工作区的循环语句拖动到组件区后成本的大小比原来不一样的问题 梁河览 何瑞 是 循环 ...
- sql_exporter的使用
sql_exporter的使用 一.背景 二.sql-exporter的使用 1.下载 2.配置文件 1.sql_exporter.yml 2.collectors 目录中的配置文件 1.collec ...
- Noip模拟33垫底反思 2021.8.8
T1 Hunter 考场上没写$%p$挂了25分.也是很牛皮,以后打完过了样例一定要检查 因为样例太小了......很容易忘记%%%% 正解随便手模就出来了. 1 #include<bits/s ...
- hdu 2586 How far away? (LCA模板)
题意: N个点,形成一棵树,边有长度. M个询问,每个询问(a,b),询问a和b的距离 思路: 模板题,看代码.DFS预处理算出每个结点离根结点的距离. 注意: qhead[maxn],而不是qhea ...
- nodejs:使用puppeteer在服务器中构建一个获取电影电视剧剧集的接口
首先我们看下数据来源: 来源于这个网站:https://z1.m1907.cn/ 可以说这个网站上能找到很多你想看的很多电影或电视剧,最重要的是很多电影电视剧在别的网站是收费的,但是在这里看是免费的, ...
- Spark记录(二):Spark程序的生命周期
本文以Spark执行模式中最常见的集群模式为例,详细的描述一下Spark程序的生命周期(YARN作为集群管理器). 1.集群节点初始化 集群刚初始化的时候,或者之前的Spark任务完成之后,此时集群中 ...
- C语言图书管理借阅系统——ncurses库的使用
一.前言 作为一只大四狗,最近还跟着大二同学修了一门课(当然不是之前没通过啦),课程是高级语言课程设计,高级语言指的是C语言 :),内容是做一个XX管理系统,我选择了图书管理系统,先介绍下我做的系统: ...