【BZOJ4071】[Apio2015]巴邻旁之桥

Description

一条东西走向的穆西河将巴邻旁市一分为二,分割成了区域 A 和区域 B。

每一块区域沿着河岸都建了恰好 1000000001 栋的建筑,每条岸边的建筑都从 0 编号到 1000000000。相邻的每对建筑相隔 1 个单位距离,河的宽度也是 1 个单位长度。区域 A 中的 i 号建筑物恰好与区域 B 中的 i 号建筑物隔河相对。
城市中有 N 个居民。第 i 个居民的房子在区域 Pi 的 Si 号建筑上,同时他的办公室坐落在 Qi 区域的 Ti 号建筑上。一个居民的房子和办公室可能分布在河的两岸,这样他就必须要搭乘船只才能从家中去往办公室,这种情况让很多人都觉得不方便。为了使居民们可以开车去工作,政府决定建造不超过 K 座横跨河流的大桥。
由于技术上的原因,每一座桥必须刚好连接河的两岸,桥梁必须严格垂直于河流,并且桥与桥之间不能相交。当政府建造最多 K 座桥之后,设 Di 表示第 i 个居民此时开车从家里到办公室的最短距离。请帮助政府建造桥梁,使得 D1+D2+⋯+DN 最小。

Input

输入的第一行包含两个正整数 K 和 N,分别表示桥的上限数量和居民的数量。

接下来 N 行,每一行包含四个参数:Pi,Si,Qi 和 Ti,表示第 i 个居民的房子在区域 Pi 的 Si 号建筑上,且他的办公室位于 Qi 区域的 Ti 号建筑上。

Output

输出仅为一行,包含一个整数,表示 D1+D2+⋯+DN 的最小值。

Sample Input

1 5
B 0 A 4
B 1 B 3
A 5 B 7
B 2 A 6
B 1 A 7

Sample Output

24

HINT

子任务

所有数据都保证:Pi 和 Qi 为字符 “A” 和 “B” 中的一个, 0≤Si,Ti≤1000000000,同一栋建筑内可能有超过 1 间房子或办公室(或二者的组合,即房子或办公室的数量同时大于等于 1)。
子任务 1 (8 分)
K=1
1≤N≤1000
子任务 2 (14 分)
K=1
1≤N≤100000
子任务 3 (9 分)
K=2
1≤N≤100
子任务 4 (32 分)
K=2
1≤N≤1000
子任务 5 (37 分)
K=2
1≤N≤100000

题解:先把不需要过桥的长度处理出来,别忘了桥的长度为1

当只有一座桥时,设位置为k,答案为∑(|ai-k|+|bi-k|),显然我们只要将ai,bi放在一起排序,取中位数就行了

当有两座桥时,我们设位置为k1,k2,那么对于第i个人,他选择kj(j=1,2)的长度为|ai-k1|+|bi-k1|,这个值有几种情况:

1.  |ai-bi|           2.  |2*kj-ai-bi|

发现他肯定会选择距离ai+bi较近的那座桥,也就是说,当我们确定了两座桥的位置后,存在一个临界值mid,使得ai+bi比mid小的都选择k1,ai+bi比mid大的都选择k2

所以我们将人按照ai+bi排序,枚举这个分界线,然后两边分别按照只有一座桥的情况处理,这就需要我们动态维护中位数+统计答案,用treap轻松搞定

注意:当可以建两座桥的时候,也要讨论只建一座桥的情况!

时间上线20s,我跑了17s,我的代码是由多丑~

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#include <cstdlib>
using namespace std;
typedef long long ll;
const int maxn=200010;
ll N,n,m,tot,r1,r2;
ll ans,tans,sum;
char sa[5],sb[5];
ll v[maxn<<2],s[maxn<<2],cnt[maxn<<2],siz[maxn<<2];
ll key[maxn<<2],ch[maxn<<2][2];
struct node
{
ll A,B;
}p[maxn];
bool cmp(node a,node b)
{
return a.A+a.B<b.A+b.B;
}
void pushup(ll x)
{
s[x]=v[x]*cnt[x]+s[ch[x][0]]+s[ch[x][1]];
siz[x]=siz[ch[x][0]]+siz[ch[x][1]]+cnt[x];
}
void rotate(ll &x,ll d)
{
ll y=ch[x][d];
ch[x][d]=ch[y][d^1],ch[y][d^1]=x;
pushup(x),pushup(y);
x=y;
}
void insert(ll &x,ll y)
{
if(!x)
{
x=++tot;
s[x]=v[x]=y,key[x]=rand(),cnt[x]=siz[x]=1;
return ;
}
siz[x]++;
if(y==v[x])
{
cnt[x]++,s[x]+=y;
return ;
}
ll d=(y>v[x]);
insert(ch[x][d],y);
if(key[ch[x][d]]>key[x]) rotate(x,d);
pushup(x);
}
void query(ll x,ll y)
{
if(!x) return ;
if(siz[ch[x][0]]>=y)
{
sum+=v[x]*cnt[x]+s[ch[x][1]];
query(ch[x][0],y);
return ;
}
if(siz[ch[x][0]]+cnt[x]>=y)
{
sum+=s[ch[x][1]]-s[ch[x][0]];
sum+=(2*siz[ch[x][0]]+cnt[x]-2*y)*v[x];
return ;
}
sum-=s[ch[x][0]]+v[x]*cnt[x];
query(ch[x][1],y-siz[ch[x][0]]-cnt[x]);
}
void del(ll &x,ll y)
{
if(v[x]==y)
{
if(cnt[x]>1)
{
cnt[x]--,siz[x]--,s[x]-=y;
return ;
}
if(ch[x][0]*ch[x][1]==0)
{
x=ch[x][0]+ch[x][1];
return ;
}
ll d=(key[ch[x][1]]>key[ch[x][0]]);
rotate(x,d),siz[x]--,s[x]-=y;
del(ch[x][d^1],y);
return ;
}
siz[x]--,s[x]-=y;
del(ch[x][y>v[x]],y);
return ;
}
int main()
{
scanf("%d%d",&m,&N);
ll a,b,i;
for(i=1;i<=N;i++)
{
scanf("%s%lld%s%lld",sa,&a,sb,&b);
if(a>b) swap(a,b);
if(sa[0]==sb[0]) tans+=b-a;
else p[++n].A=a,p[n].B=b;
}
tans+=n;
sort(p+1,p+n+1,cmp);
if(m==1)
{
for(i=1;i<=n;i++) insert(r1,p[i].A),insert(r1,p[i].B);
query(r1,n);
printf("%lld",tans+sum);
return 0;
}
for(i=1;i<=n;i++) insert(r2,p[i].A),insert(r2,p[i].B);
sum=0; query(r2,n);
ans=tans+sum;
for(i=1;i<=n;i++)
{
del(r2,p[i].A),del(r2,p[i].B);
insert(r1,p[i].A),insert(r1,p[i].B);
sum=0;
query(r1,i),query(r2,n-i);
ans=min(ans,tans+sum);
}
printf("%lld",ans);
return 0;
}

【BZOJ4071】[Apio2015]巴邻旁之桥 Treap的更多相关文章

  1. [bzoj4071] [Apio2015]巴邻旁之桥

    Description 一条东西走向的穆西河将巴邻旁市一分为二,分割成了区域 A 和区域 B. 每一块区域沿着河岸都建了恰好 1000000001 栋的建筑,每条岸边的建筑都从 0 编号到 10000 ...

  2. BZOJ4071 & 洛谷3644 & UOJ112:[APIO2015]巴邻旁之桥——题解

    https://www.lydsy.com/JudgeOnline/problem.php?id=4071 https://www.luogu.org/problemnew/show/P3644 ht ...

  3. 4071: [Apio2015]巴邻旁之桥

    Description 一条东西走向的穆西河将巴邻旁市一分为二,分割成了区域 A 和区域 B. 每一块区域沿着河岸都建了恰好 1000000001 栋的建筑,每条岸边的建筑都从 0 编号到 10000 ...

  4. [BZOJ4071][APIO2015]八邻旁之桥

    BZOJ(这题是BZOJ权限题,有权限号的就去看看吧) Luogu(良心洛谷) 题目描述 一条东西走向的穆西河将巴邻旁市一分为二,分割成了区域\(A\)和区域\(B\). 每一块区域沿着河岸都建了恰好 ...

  5. [APIO2015]巴邻旁之桥

    Bzoj权限题 luogu题面 先去掉同边的 首先k==1,即求一个点j 使\(\sum_{i\in A} |D_i - D_j| + \sum_{i\in B} |D_i - D_j|\)最小 因为 ...

  6. bzoj 4071: [Apio2015]巴邻旁之桥【splay】

    用权值线段树会容易一些并快一些,但是想复健一下splay所以打了splay 然后果然不会打了. 解题思路: 首先把家和办公室在同一侧的提出来直接加进答案里: 对于k=1,直接选所有办公室和家的中位数即 ...

  7. [APIO2015]八邻旁之桥——非旋转treap

    题目链接: [APIO2015]八邻旁之桥 对于$k=1$的情况: 对于起点和终点在同侧的直接计入答案:对于不在同侧的,可以发现答案就是所有点坐标与桥坐标的差之和+起点与终点不在同一侧的人数. 将所有 ...

  8. 洛谷 P3644 [APIO2015]八邻旁之桥 解题报告

    P3644 [APIO2015]八邻旁之桥 题目描述 一条东西走向的穆西河将巴邻旁市一分为二,分割成了区域\(A\)和区域\(B\). 每一块区域沿着河岸都建了恰好\(1000000001\)栋的建筑 ...

  9. 【BZOJ4071】八邻旁之桥(线段树)

    [BZOJ4071]八邻旁之桥(线段树) 题面 BZOJ权限题,洛谷链接 题解 既然\(k<=2\) 那么,突破口就在这里 分类讨论 ①\(k=1\) 这...不就是中位数吗.... 直接把所有 ...

随机推荐

  1. 【剑指Offer面试题】 九度OJ1510:替换空格

    c/c++ 中的字符串以"\0"作为结尾符.这样每一个字符串都有一个额外字符的开销. 以下代码将造成内存越界. char str[10]; strcpy(str, "01 ...

  2. 微信小程序请求wx.request数据,渲染到页面

    先说一下基本使用.官网也有. 比如说你在App.js里面有这些变量.想修改某些值. data: { main_view_bgcolor: "", border: "&qu ...

  3. Bootloader和Root

    1. Unlock Bootloader是解除系统启动加载器(Bootloader)的原厂限制, 让用户可以使用到更多的功能(如刷新内核.刷ROM.修改超频....) Bootloader(系统启动加 ...

  4. Makefile 8——使用依赖关系文件

    Makefile中存在一个include指令,它的作用如同C语言中的#include预处理指令.在Makefile中,可以通过include指令将自动生成的依赖关系文件包含进来,从而使得依赖关系文件中 ...

  5. poj2559单调栈

    题意:给出连续的矩形的高....求最大面积 #include<iostream> #include<stack> #include<stdio.h> using n ...

  6. Unix系统编程()原子操作和竞争条件

    竞争状态是这样一种情形:操作共享资源的两个进程(或线程),其结果取决于一个无法预期的顺序,即这些进程获得CPU使用权的先后相对顺序. 以独占的方式创建一个文件 当同时指定了O_EXCL和O_CREAT ...

  7. 科技巨头们以 "A" 取名的时尚潮流

    科技巨头们以 "A" 取名的时尚潮流 from 公众号  WebHub  世界上有许多巨头公司喜欢以字母 a 打头作公司起名.改名,这主要是因为电话薄是以字母排序的(外国人习惯家里 ...

  8. eclipse不能自动编译生成class文件的解决办法

    最近在项目项目开发过程中遇到eclipse不能自动编译生成class文件,当时很纳闷,每次修改代码后运行都是修改前的效果,没辙了,只好反编译原来的class文件,结果发现,class文件里并没有看到修 ...

  9. 应用于Java中的一个开源的表达式语言(Expression Language)

    OGNL(英文全称:Object Graph Navigation Language,中文名:对象导航图语言)是应用于Java中的一个开源的表达式语言(Expression Language),它被集 ...

  10. ResultSet是结果集对象

    ResultSet是结果集对象 DriverManager管理一组驱动程序 PreparedStatement预编译的,用来发送和执行SQL语句的