【BZOJ4071】八邻旁之桥(线段树)
【BZOJ4071】八邻旁之桥(线段树)
题面
题解
既然\(k<=2\)
那么,突破口就在这里
分类讨论
①\(k=1\)
这。。。不就是中位数吗。。。。
直接把所有起点重点排个序,
算下中位数就行啦
②\(k=2\)
似乎不好搞了
orz ZSY Dalao
我太弱了
我就是一个Vegetable Chicken
ZSY看一眼就会做
补充:ZSY大佬提醒我,Bridge我写错了
所以:#define Brige Bridge
首先,我们来看一看,如果有两座桥,
一个人会怎么动呢?
如果桥在他所移动的横向区间内
那么,一定会过这座桥,距离为\(dis(Qi-Ti)+1\)
如果,没有桥在他的区间内
他就要先走到桥,再从桥走过来
此时距离为\(abs(Qi-Brige)+abs(Ti-Brige)+1\)
这个东西再结合图像化个简
等于\(2abs(\frac{Qi+Ti}{2}-Brige)+1\)
所以,这个人走的桥一定是离\(\frac{Qi+Ti}{2}\)较近的桥
因此,把所有人按照\(\frac{Qi+Ti}{2}\)排序之后
开始枚举在哪个位置割开
然后左边的都走左边的桥
右边的都走右边的桥
拆成了两边之后就是\(k=1\)的情况了
但是因为是动态维护区间的中位数
所以要找个东西来维护
权值线段树,平衡树都是可以的
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<set>
#include<map>
#include<vector>
#include<queue>
using namespace std;
#define ll long long
#define MAX 250000
#define lson (now<<1)
#define rson (now<<1|1)
inline int read()
{
int x=0,t=1;char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=-1,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return x*t;
}
struct Node{int a,b;}w[MAX];
bool operator<(Node a,Node b){return a.a+a.b<b.a+b.b;}
int cnt;
long long ans;
int K,n,a[MAX],b[MAX],S[MAX],top;
struct SegMentTree
{
struct node
{
int size;ll sum;
}t[MAX<<2];
int Kth(int now,int l,int r,int k)
{
if(l==r)return l;
int mid=(l+r)>>1;
if(k<=t[lson].size)return Kth(lson,l,mid,k);
else return Kth(rson,mid+1,r,k-t[lson].size);
}
void Modify(int now,int l,int r,int pos,int ww)
{
t[now].size+=ww;t[now].sum+=1ll*ww*S[pos];
if(l==r)return;
int mid=(l+r)>>1;
if(pos<=mid)Modify(lson,l,mid,pos,ww);
else Modify(rson,mid+1,r,pos,ww);
}
ll QueryV(int now,int l,int r,int al,int ar)
{
if(al<=l&&r<=ar)return t[now].sum;
int mid=(l+r)>>1;ll ret=0;
if(al<=mid)ret+=QueryV(lson,l,mid,al,ar);
if(ar>mid)ret+=QueryV(rson,mid+1,r,al,ar);
return ret;
}
int QueryS(int now,int l,int r,int al,int ar)
{
if(al<=l&&r<=ar)return t[now].size;
int mid=(l+r)>>1,ret=0;
if(al<=mid)ret+=QueryS(lson,l,mid,al,ar);
if(ar>mid)ret+=QueryS(rson,mid+1,r,al,ar);
return ret;
}
}T[2];
int main()
{
K=read();n=read();
char ch[2];
if(K==1)
{
int tot=0;
for(int i=1;i<=n;++i)
{
scanf("%s",ch);int p=ch[0]-'A';
int s=read();
scanf("%s",ch);int q=ch[0]-'A';
int t=read();
if(p==q){ans+=abs(s-t);continue;}
else if(p==1)swap(s,t);
++tot;a[tot]=s;b[tot]=t;S[++top]=s;S[++top]=t;
}
sort(&S[1],&S[top+1]);
int G=S[top/2];
for(int i=1;i<=top;++i)ans+=abs(G-S[i]);
printf("%lld\n",ans+tot);
return 0;
}
for(int i=1;i<=n;++i)
{
scanf("%s",ch);int p=ch[0]-'A';
int s=read();
scanf("%s",ch);int q=ch[0]-'A';
int t=read();
if(p==q){ans+=abs(s-t);continue;}
ans++;S[++top]=s;S[++top]=t;
if(s>t)swap(s,t);
w[++cnt]=(Node){s,t};
}
if(!cnt){printf("%lld\n",ans);return 0;}
sort(&w[1],&w[cnt+1]);
sort(&S[1],&S[top+1]);
top=unique(&S[1],&S[top+1])-S-1;
for(int i=1;i<=cnt;++i)
{
w[i].a=lower_bound(&S[1],&S[top+1],w[i].a)-S;
w[i].b=lower_bound(&S[1],&S[top+1],w[i].b)-S;
T[1].Modify(1,1,top,w[i].a,1);
T[1].Modify(1,1,top,w[i].b,1);
}
long long sum=1e18;
for(int i=1;i<=cnt;++i)
{
T[0].Modify(1,1,top,w[i].a,1);
T[0].Modify(1,1,top,w[i].b,1);
T[1].Modify(1,1,top,w[i].a,-1);
T[1].Modify(1,1,top,w[i].b,-1);
int p1=T[0].Kth(1,1,top,i);//找中位数
int p2=T[1].Kth(1,1,top,cnt-i);
long long D0=0;
D0+=1ll*T[0].QueryS(1,1,top,1,p1)*S[p1]-T[0].QueryV(1,1,top,1,p1);
D0+=T[0].QueryV(1,1,top,p1,top)-1ll*T[0].QueryS(1,1,top,p1,top)*S[p1];
long long D1=0;
D1+=1ll*T[1].QueryS(1,1,top,1,p2)*S[p2]-T[1].QueryV(1,1,top,1,p2);
D1+=T[1].QueryV(1,1,top,p2,top)-1ll*T[1].QueryS(1,1,top,p2,top)*S[p2];
sum=min(sum,D0+D1);
}
printf("%lld\n",ans+sum);
return 0;
}
【BZOJ4071】八邻旁之桥(线段树)的更多相关文章
- 洛谷 P3644 [APIO2015]八邻旁之桥 解题报告
P3644 [APIO2015]八邻旁之桥 题目描述 一条东西走向的穆西河将巴邻旁市一分为二,分割成了区域\(A\)和区域\(B\). 每一块区域沿着河岸都建了恰好\(1000000001\)栋的建筑 ...
- [APIO2015]八邻旁之桥——非旋转treap
题目链接: [APIO2015]八邻旁之桥 对于$k=1$的情况: 对于起点和终点在同侧的直接计入答案:对于不在同侧的,可以发现答案就是所有点坐标与桥坐标的差之和+起点与终点不在同一侧的人数. 将所有 ...
- [luoguP3644] [APIO2015]八邻旁之桥(权值线段树)
传送门 首先如果起点终点都在同一侧可以直接处理,如果需要过桥答案再加1 对于k等于1的情况 桥的坐标为x的话,a和b为起点和终点坐标 $ans=\sum_{1}^{n} abs(a_{i}-x)+ab ...
- [BZOJ4071][APIO2015]八邻旁之桥
BZOJ(这题是BZOJ权限题,有权限号的就去看看吧) Luogu(良心洛谷) 题目描述 一条东西走向的穆西河将巴邻旁市一分为二,分割成了区域\(A\)和区域\(B\). 每一块区域沿着河岸都建了恰好 ...
- [APIO2015]八邻旁之桥
题面在这里 sol 这是一个\(Splay\)的题解 首先,如果一个人的家和办公室在同一侧,我们可以直接预处理; 如果不在同一侧,也可以加上1(当然要过桥啦) 当k==1时 我们设桥的位置为\(pos ...
- 题解【luoguP3644 [APIO2015]八邻旁之桥】
题目链接 题解 家和公司在同侧 简单,直接预处理掉 若 \(k=1\) 取所有的居民的\(\frac{家坐标+公司坐标}{2}\)的所有坐标的正中间建一座桥,使所有居民到的距离最小. 实现方法:线段树 ...
- APIO2015 八邻旁之桥/巴邻旁之桥
题目描述: bz luogu 题解: 贪心+权值线段树. $K=1$的时候,答案为$\sum |x-l| + |x-r|$,所以所有端点排序后取中位数即可. $K=2$的时候,一定是左边的一些走左边的 ...
- 洛谷 P3644 [APIO2015]八邻旁之桥(对顶堆维护中位数)
题面传送门 题意: 一条河将大地分为 \(A,B\) 两个部分.两部分均可视为一根数轴. 有 \(n\) 名工人,第 \(i\) 名的家在 \(x_i\) 区域的 \(a_i\) 位置,公司在 \(y ...
- 【BZOJ4071】[Apio2015]巴邻旁之桥 Treap
[BZOJ4071][Apio2015]巴邻旁之桥 Description 一条东西走向的穆西河将巴邻旁市一分为二,分割成了区域 A 和区域 B. 每一块区域沿着河岸都建了恰好 1000000001 ...
随机推荐
- 网络编程之socketserver
网络编程之socketserver """ socketserver.py 中的5个基础类 +------------+ | BaseServer | +-------- ...
- "abc123 ,def456",反转字母,其他位置不变
"abc123 ,def456",反转字母,其他位置不变. 无意间看到个有意思的面试题,忽然来了兴趣想着来做一下. 操作字符串用正则的效率比较高,但第一反应还是用原生来操作.下面说 ...
- Jmeter_上传与下载
今天重点说一下Jmeter的上传与下载 1:Jmeter上传文件 首先确认你的文件名称,参数名称,MIME类型,这些可以从接口文档里面获取,或者直接在页面抓包然后从请求头里面查看.注意,此处我的文件路 ...
- laravel服务容器-----深入理解控制反转(IoC)和依赖注入(DI)
首先大家想一想什么是容器,字面意思就是盛放东西的东西,常见的变量,对象属性都是容器,一个容器能够装什么东西,完全在于你对这个容器的定义.有的容器不仅仅只是存文本,变量,而是对象,属性,那么我们通过这种 ...
- win7(windows 7)系统下安装SQL2005(SQL Server 2005)图文教程
操作系统:Microsoft Windows 7 旗舰版(32位) 数据库版本:SQL Server 2005 简体中文开发板 数据库下载链接: https://pan.baidu.com/s/1cq ...
- html拨打电话、发送短信、发送邮件的链接写法
拨打电话 <a href="tel:88888888">呼叫</a> 发送短信 <a href="sms:88888888"> ...
- python进阶学习笔记(一)
python进阶部分要学习的内容: 学习目标: 1.函数式编程 1.1,什么是函数式编程 函数式编程是一种抽象计算的编程模式 不同语言的抽象层次不同: 函数式编程的特点: python支持的函数式编程 ...
- STAThread 和 MTAThread
STAThread:single threaded apartment 直译过来是:单线程单元套间 MTAThread:multiple threaded apartment 直译过来是:多线程单元套 ...
- LNMP Yii2 验证码不显示问题最终解决方案
首先,本地使用OK! 然后,新配置的LNMP环境,验证码一直显示不出来,看了Yii2的验证码存在session里,怀疑是session有问题. 在测试其他页面的时候,发现:session_start( ...
- Ubuntu下Maven配置与Maven项目创建教程
一. Ubuntu下Maven配置 windows下Maven配置参考http://www.cnblogs.com/LexMoon/p/JavaMaven.html ubuntu下Maven地址htt ...