用权值线段树会容易一些并快一些,但是想复健一下splay所以打了splay

然后果然不会打了。

解题思路:

首先把家和办公室在同一侧的提出来直接加进答案里;

对于k=1,直接选所有办公室和家的中位数即可,因为显然如果选的点两边的办公室和家的点不一样多的话,显然可以向多的一侧移动来使得总路程减少;

对于k=2,所有人按家和办公室的中点排序,然后显然有一个断点,左边的人走左边的桥,右边的人走右边的桥。因为选择距离中点近的桥一定不会变差。然后枚举这个断点,先从左到右依次加进去,每次加完之后把当前在splay里的这部分答案(中位数和其他数的差得绝对值)存起来,然后从左到右依次删掉,此时splay里的是除了1~i的部分,把这部分的答案加上之前存起来的答案就是最后答案,然后在这些答案里取min即可。

!这里因为取min所以初始值设为inf了,注意特判一下没有家和办公室在两岸的人把初始值改成0

splay注意事项:如果涉及插入,要先插两个辅助点inf,-inf进去,涉及删除则要插四个;

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<map>
using namespace std;
const int N=200005;
const long long inf=1e15;
int k,n,top,root,tot;
long long q[N],ans;
struct qwe
{
int s,t,m;
qwe(int S=0,int T=0,int M=0)
{
s=S,t=T,m=M;
}
}a[N];
struct Splay
{
int f,c[2],s;
long long sum,v;
}t[N];
inline bool cmp(const qwe &a,const qwe &b)
{
return a.m<b.m;
}
int read()
{
int r=0,f=1;
char p=getchar();
while(p>'9'||p<'0')
{
if(p=='-')
f=-1;
p=getchar();
}
while(p>='0'&&p<='9')
{
r=r*10+p-48;
p=getchar();
}
return r*f;
}
inline void ud(int x)
{
t[x].s=t[t[x].c[0]].s+t[t[x].c[1]].s+1;
t[x].sum=t[t[x].c[0]].sum+t[t[x].c[1]].sum+(long long)t[x].v;
}
void zhuan(int x,int &k)
{
int y=t[x].f,z=t[y].f,l=t[y].c[0]!=x,r=l^1;
if(y!=k)
t[z].c[t[z].c[0]!=y]=x;
else
k=x;
t[x].f=z;
t[y].c[l]=t[x].c[r];
t[t[y].c[l]].f=y;
t[x].c[r]=y;
t[y].f=x;
ud(y);
ud(x);
}
void splay(int x,int &k)
{
while(x!=k)
{
int y=t[x].f,z=t[y].f;
if(y!=k)
{
if((t[y].c[0]==x)^(t[z].c[0]==y))
zhuan(x,k);
else
zhuan(y,k);
}
zhuan(x,k);
}
}
int paiming(int x,int va)
{
int r=0;
while(x)
{
if(t[x].v>va)
x=t[x].c[0];
else
r+=t[t[x].c[0]].s+1,x=t[x].c[1];
}
return r;
}
int shu(int x,int k)
{
if(!x)
return 0;
if(t[t[x].c[0]].s+1==k)
return x;
else if(t[t[x].c[0]].s>=k)
return shu(t[x].c[0],k);
else
return shu(t[x].c[1],k-t[t[x].c[0]].s-1);
}
void charu(int k)
{
int x=shu(root,paiming(root,k));
splay(x,root);
int y=shu(t[x].c[1],1);
splay(y,t[x].c[1]);
t[y].c[0]=++tot;
t[tot].f=y;
t[tot].v=k;
t[tot].sum=k;
t[tot].s=1;
ud(y);
ud(x);
}
void shanchu(int k)
{
int x=shu(root,paiming(root,k)-1);//cerr<<k<<" "<<paiming(root,k)<<endl;
splay(x,root);
int y=shu(t[x].c[1],2);
splay(y,t[x].c[1]);
t[t[y].c[0]].f=0;
t[y].c[0]=0;
ud(y);
ud(x);
}
long long ques(int x)
{//cerr<<t[x].v<<" ";
splay(x,root);
return 1ll*t[x].v*(t[t[x].c[0]].s-2)-t[t[x].c[0]].sum+t[t[x].c[1]].sum-1ll*t[x].v*(t[t[x].c[1]].s-2)-4*inf;
}
int main()
{
k=read(),n=read();
for(int i=1;i<=n;i++)
{
char c1=0,c2=0;
while(c1!='A'&&c1!='B')
c1=getchar();
int x=read();
while(c2!='A'&&c2!='B')
c2=getchar();
int y=read();//cerr<<c1<<" "<<c2<<" "<<x<<" "<<y<<endl;
if(c1==c2)
ans+=abs(y-x);
else
a[++top]=qwe(x,y,abs(x+y)>>1);
}
n=top;
if(k==1)
{
int e[N];
for(int i=1;i<=n;i++)
e[i]=a[i].s,e[i+n]=a[i].t;
sort(e+1,e+1+n*2);
int p=e[n];
for(int i=1;i<=n*2;i++)
ans+=1ll*abs(e[i]-p);
printf("%lld\n",ans+n);
return 0;
}
sort(a+1,a+1+n,cmp);
tot=4;
t[1].v=-inf,t[1].s=4,t[1].c[0]=2,t[1].c[1]=3,root=1;
t[2].v=-inf,t[2].sum=-inf,t[2].s=1,t[2].f=1;
t[3].v=inf,t[3].sum=2ll*inf,t[3].s=2,t[3].f=1,t[3].c[1]=4;
t[4].v=inf,t[4].sum=inf,t[4].s=1,t[4].f=3;
for(int i=1;i<=n;i++)
{
charu(a[i].s),charu(a[i].t);
q[i]=ques(shu(root,i+2));//cerr<<q[i]<<endl;
}//cerr<<"ok"<<endl;
long long sum=inf;
if(!n)
sum=0;
for(int i=1;i<=n;i++)
{
shanchu(a[i].s),shanchu(a[i].t);
sum=min(sum,q[i]+ques(shu(root,n-i+2)));
}
printf("%lld\n",ans+sum+n);
return 0;
}

bzoj 4071: [Apio2015]巴邻旁之桥【splay】的更多相关文章

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

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

  2. 【BZOJ4071】[Apio2015]巴邻旁之桥 Treap

    [BZOJ4071][Apio2015]巴邻旁之桥 Description 一条东西走向的穆西河将巴邻旁市一分为二,分割成了区域 A 和区域 B. 每一块区域沿着河岸都建了恰好 1000000001 ...

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

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

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

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

  5. [APIO2015]巴邻旁之桥

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

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

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

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

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

  8. 【BZOJ4071】【APIO2015】巴邻旁之桥

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

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

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

随机推荐

  1. Unique Paths II (dp题)

    Follow up for "Unique Paths": Now consider if some obstacles are added to the grids. How m ...

  2. Meteor Blaze

    Blaze是Meteor 软件包用于构建现场反应模板. Render方法 这种方法被用于绘制模板到DOM.首先,我们将创建 myNewTemplate 之后渲染. 我们增加 myContainer 这 ...

  3. Antivius for Linux

    http://www.clamav.net/   https://www.avast.com/zh-cn/linux-server-antivirus   http://www.f-prot.com/ ...

  4. iOS设计模式 - (1)概述

    近期可自由安排的时间比較多, iOS应用方面, 没什么好点子, 就先放下, 不写了.花点时间学学设计模式. 之后将会写一系列博文, 记录设计模式学习过程. 当然, 由于我自己是搞iOS的, 所以之后设 ...

  5. Office文档如何转换 PDF 转 DOC XLS

    1 使用Adobe Acrobat Pro,打开任意PDF都可以转换为XLSX格式(似乎没找到XLS)   2 如果你转换之后的东西无法打开,则先转换成DOC,然后再把DOC全选复制粘贴到XLS即可 ...

  6. 积跬步,聚小流------ps有用小技巧,改变png图标颜色

    *  实现效果: 原图:  改动后: *  实现目的: 满足为实现不同界面色彩搭配改动png图标的颜色 *  实现方法: 1.打开Photoshop工具,导入须要进行改动的png图标: 2.对导入的图 ...

  7. ==和equals的差别

    == 和 Equals 的差别 1. == 是一个运算符. 2.Equals则是string对象的方法.能够.(点)出来. 我们比較无非就是这两种 1.基本数据类型比較 2.引用对象比較 1.基本数据 ...

  8. 【iOS系列】-oc中特有的语法

    [iOS系列]-oc中特有的语法 oc数据类型: 1,基本类型 2,对象类型 3,id 4,BOOL 5,block 6,SEL 1:category 使用继承关系来扩充一个类,有一个弊病,高耦合性 ...

  9. 安装mint的问题集锦

    1.修改DNS解析配置 刚刚安装完mint可能现无法连接源的问题,总是说dns解析错误,这个可能是dns配置文件造成的,因为官网下的mint很可能是配置了国外的dns解析,比如我刚安上时,就是默认配置 ...

  10. HBase在大搜车金融业务中的应用实践

    摘要: 2017云栖大会HBase专场,大搜车高级数据架构师申玉宝带来HBase在大搜车金融业务中的应用实践.本文主要从数据大屏开始谈起,进而分享了GPS风控实践,包括架构.聚集分析等,最后还分享了流 ...