Time Limit: 30 Sec  Memory Limit: 256 MB

Description

  你有n辆车,分别a1, a2, ..., an位置和n个加油站,分别在b1, b2, ... ,bn 。每个加油站只能支持一辆车的加油,所以你要把这些车开到不同的加油站加油。一个车从x位置开到y位置的代价为 |x-y| ,问如何安排车辆,使得代价之和最小。同时你有q个操作,每次操作会修改第i辆车的位置到x,你要回答每次修改操作之后最优安排方案的总代价。

Input

  第一行一个正整数n,接下来一行n个整数a1, a2, ...,an,接下来一行n个整数b1, b2,... ,bn。
  接下来一行一个正整数q,表示操作的个数。
  接下来q行,每行有两个整数i(1 ≤ i ≤ n)和x,表示将i这辆车开到x位置的操作。
  1 ≤ n, q ≤ 5 * 10^4,所有的车和加油站的范围一直在0到10^9之间。

Output

  共q+1行,第一行表示一开始的最优代价。接下来q行,第i行表示操作i之后的最优代价。

Sample Input

  2
  1 2
  3 4
  1
  1 3

Sample Output

  4
  2
  【样例解释】
  一开始将第一辆车开到位置4,将第二辆车开到位置3,代价为 |4-1|+|3-2|=4。
  修改后第一辆车的位置变成3,代价为 |3-3|+|4-2|=2。

Solution

  一个显然的结论是把a和b排序后按顺序匹配最优,这里就不做说明了。假设我们有一种模拟这个匹配的过程如下:从左到右扫,碰到车就把就把车加入队列,碰到加油站也把加油站加入队列,如果有车和加油站同时在队列中,我们让他们两两匹配,无法匹配我们就拖着所有车或加油站一起向右走。按照这个思路,我们可以这么计算答案:把坐标离散化后,每个车往对应坐标+1,加油站往对应坐标-1,把+1和-1的标记做前缀和,每个坐标的前缀和的绝对值乘上到下一个坐标的距离之和就是答案。那么一次修改就是把这些标记区间+1或-1,同时维护答案。这个可以分块搞,做法如下:按坐标顺序把前缀和分成若干块,每块内把前缀和排序,并求出排序后的每个坐标到排序前的下一个坐标的距离的前缀和,修改时在边缘的块直接暴力重构,剩下整块修改时,标记前缀和大于0和小于0的分开统计答案,排序可以基排桶排啥的,0的分界点也可以动态维护,但我懒所以都写的log,还好能过。总复杂度$O(nlogn\sqrt{n})$或$O(n\sqrt{n})$。

Code

#include<cstdio>
#include<algorithm>
using namespace std;
#define ll long long
inline int read()
{
int x;char c;
while((c=getchar())<''||c>'');
for(x=c-'';(c=getchar())>=''&&c<='';)x=x*+c-'';
return x;
}
#define MN 50000
#define MP 150000
#define K 400
int a[MN+],b[MN+],c[MN+],d[MN+];
int p[MP+],pn,s[MP+],ad[K+];
ll ans;
struct data{int s;ll p;}t[MP+];
bool cmp(const data&a,const data&b){return a.s<b.s;}
inline ll z(int x){return x<?-x:x;}
void build(int x)
{
int i;
for(i=x;i<x+K;++i)t[i].s=s[i],t[i].p=p[i];
sort(t+x,t+x+K,cmp);
for(i=x;++i<x+K;)t[i].p+=t[i-].p;
}
void add(int l,int r,int x)
{
int i,lk=(l-)/K,rk=(r-)/K;
for(i=lk*K+;i<=(lk+)*K;++i)s[i]+=ad[lk];ad[lk]=;
for(i=l;i<=r&&i<=(lk+)*K;++i)ans-=z(s[i])*p[i],ans+=z(s[i]+=x)*p[i];
build(lk*K+);
for(i=lk+;i<rk;++i)
{
ad[i]+=x;
int l=i*K+,r=(i+)*K,mid,res=;
if(x<)
{
while(l<=r)
if(t[mid=l+r>>].s+ad[i]<)res=mid,l=mid+;
else r=mid-;
ans+=*t[res].p-t[(i+)*K].p;
}
else
{
while(l<=r)
if(t[mid=l+r>>].s+ad[i]<=)res=mid,l=mid+;
else r=mid-;
ans-=*t[res].p-t[(i+)*K].p;
}
}
if(lk==rk)return;
for(i=rk*K+;i<=(rk+)*K;++i)s[i]+=ad[rk];ad[rk]=;
for(i=r;i>rk*K;--i)ans-=z(s[i])*p[i],ans+=z(s[i]+=x)*p[i];
build(rk*K+);
}
int main()
{
int n=read(),q,i,j;
for(i=;i<=n;++i)p[++pn]=a[i]=read();
for(i=;i<=n;++i)p[++pn]=b[i]=read();
for(q=read(),i=;i<=q;++i)c[i]=read(),p[++pn]=d[i]=read();
sort(p+,p+pn+);
for(i=,j=;i<=pn;++i)if(p[i]!=p[j])p[++j]=p[i];
for(i=j;++i<=pn;)p[i]=;pn=j;
for(i=;i<=n;++i)
++s[a[i]=lower_bound(p+,p+pn+,a[i])-p],
--s[b[i]=lower_bound(p+,p+pn+,b[i])-p];
for(i=;i<=q;++i)d[i]=lower_bound(p+,p+pn+,d[i])-p;
for(i=;i<pn;++i)ans+=z(s[i]+=s[i-])*(p[i]=p[i+]-p[i]);p[i]=;
printf("%lld\n",ans);
for(i=;i<=pn;i+=K)build(i);
for(i=;i<=q;++i)
{
if(a[c[i]]<d[i])add(a[c[i]],d[i]-,-);
else if(d[i]<a[c[i]])add(d[i],a[c[i]]-,);
a[c[i]]=d[i];
printf("%lld\n",ans);
}
}

[BZOJ]4908: [BeiJing2017]开车的更多相关文章

  1. 【BZOJ4908】[BeiJing2017]开车 分块

    [BZOJ4908][BeiJing2017]开车 Description 你有n辆车,分别a1, a2, ..., an位置和n个加油站,分别在b1, b2, ... ,bn .每个加油站只能支持一 ...

  2. [bzoj4908][BeiJing2017]开车

    来自FallDream的博客,未经允许,请勿转载,谢谢. 你有n辆车,分别a1, a2, ..., an位置和n个加油站,分别在b1, b2, ... ,bn .每个加油站只能支持一辆车的加油,所以你 ...

  3. bzoj 4860 [BeiJing2017]树的难题

    题面 https://www.lydsy.com/JudgeOnline/problem.php?id=4860 题解 点分治 设当前重心为v 假设已经把所有边按照出发点第一关键字, 颜色第二关键字排 ...

  4. 【BZOJ】4861: [Beijing2017]魔法咒语 AC自动机+DP+矩阵快速幂

    [题意]给定n个原串和m个禁忌串,要求用原串集合能拼出的不含禁忌串且长度为L的串的数量.(60%)n,m<=50,L<=100.(40%)原串长度为1或2,L<=10^18. [算法 ...

  5. 【BZOJ】【2752】【HAOI2012】高速公路(Road)

    数学期望/线段树 然而又是一道road= =上一道是2750…… 下次不要一看期望题就弃疗么…… 期望题≠不可做题……!! 其实在这题中,期望就是(所有情况下 权值之和)/(总方案数) 因为是等概率抽 ...

  6. BZOJ 2127: happiness [最小割]

    2127: happiness Time Limit: 51 Sec  Memory Limit: 259 MBSubmit: 1815  Solved: 878[Submit][Status][Di ...

  7. BZOJ 3275: Number

    3275: Number Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 874  Solved: 371[Submit][Status][Discus ...

  8. BZOJ 2879: [Noi2012]美食节

    2879: [Noi2012]美食节 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 1834  Solved: 969[Submit][Status] ...

  9. bzoj 4610 Ceiling Functi

    bzoj 4610 Ceiling Functi Description bzoj上的描述有问题 给出\(n\)个长度为\(k\)的数列,将每个数列构成一个二叉搜索树,问有多少颗形态不同的树. Inp ...

随机推荐

  1. 敏捷冲刺每日报告——Day1

    1.情况简述 Alpha阶段第一次Scrum Meeting 敏捷开发起止时间 2017.10.25 00:00 -- 2017.10.26 00:00 讨论时间地点 2017.10.25晚9:30, ...

  2. transient 与 volatile 笔记

    1. transient 词义:瞬间的,短暂的 首先说说"序列化",把一个对象的表示转化为字节流的过程称为串行化(也称为序列化,serialization),从字节流中把对象重建出 ...

  3. 个人作业2:QQ音乐APP案例分析

    APP案例分析 QQ音乐 选择理由:毕竟作为QQ音乐九年的资深老用户以及音乐爱好者 第一部分 调研 1.第一次上手的体验    我算是很早期的QQ音乐的用户,用QQ音乐七八年,除了体验各方面还不错之外 ...

  4. django BBS

    https://github.com/triaquae/py_training/tree/master/OldboyBBS2 http://www.cnblogs.com/zhming26/p/592 ...

  5. 前端之bootstrap模态框

    简介:模态框(Modal)是覆盖在父窗体上的子窗体.通常,目的是显示来自一个单独的源的内容,可以在不离开父窗体的情况下有一些互动.子窗体可提供信息.交互等. Modal简介 Modal实现弹出表单 M ...

  6. nyoj 移位密码

    移位密码 时间限制:1000 ms  |  内存限制:65535 KB 难度:0   描述 移位密码是最简单的一类代替密码,具体算法就是将字母表的字母右移k个位置(k<26),并对字母表长度作模 ...

  7. 集合Collection总览

    前言 声明,本文使用的是JDK1.8 从今天开始正式去学习Java基础中最重要的东西--->集合 无论在开发中,在面试中这个知识点都是非常非常重要的,因此,我在此花费的时间也是很多,得参阅挺多的 ...

  8. windows 10下通过python3.6成功搭建jupyter 服务器

    最近通过python学习爬虫技术,发现一个工具jupyter notebook很不错,该工具明显优势通过浏览器可以输入多行python代码,支持在线运行以及运行结果保存功能,在线验证python小模块 ...

  9. Python struct模块

    有的时候需要用python处理二进制数据,比如,存取文件,socket操作时.这时候,可以使用python的struct模块来完成.可以用 struct来处理c语言中的结构体. struct模块中最重 ...

  10. maven常见问题处理(3-4)配置代理服务器

    有的公司基于安全因素考虑,要求员工使用通过安全认证的代理访问因特网. 这时就需要为Maven配置HTTP代理. 在目录~/.m2/setting.xml文件中编辑如下(如果没有该文件,则复制$M2_H ...