APIO2015 八邻旁之桥/巴邻旁之桥
题目描述:
题解:
贪心+权值线段树。
$K=1$的时候,答案为$\sum |x-l| + |x-r|$,所以所有端点排序后取中位数即可。
$K=2$的时候,一定是左边的一些走左边的桥,右边的一些走右边的桥。
问题是按什么顺序排序。
答案是按线段中点排序。
原因是,对于河两岸的一对点和两座桥,选择的一定是离线段中点近的那个。
考虑如何快速计算答案,我们可以用权值线段树维护区间和与中位数。(当然也可以用平衡树)
代码:
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long ll;
const int N = 100050;
const int M = 100*N;
template<typename T>
inline void read(T&x)
{
T f = 1,c = 0;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9'){c=c*10+ch-'0';ch=getchar();}
x = f*c;
}
int K,n;
ll ans;
char op1[2],op2[2];
struct Point
{
int l,r;
Point(){}
Point(int l,int r):l(l),r(r){}
bool operator < (const Point&a)const{return l+r<a.l+a.r;}
}p[N];
int s[N<<1],tl;
const int inf = 2000000001;
struct segtree
{
int rt,tot,ls[M],rs[M],siz[M];
ll sum[M];
void insert(int l,int r,int&u,int qx)
{
if(!u)u=++tot;siz[u]++,sum[u]+=qx;
if(l==r)return ;
int mid = (l+r)>>1;
if(qx<=mid)insert(l,mid,ls[u],qx);
else insert(mid+1,r,rs[u],qx);
}
void erase(int l,int r,int u,int qx)
{
siz[u]--,sum[u]-=qx;if(l==r)return ;
int mid = (l+r)>>1;
if(qx<=mid)erase(l,mid,ls[u],qx);
else erase(mid+1,r,rs[u],qx);
}
int query(int l,int r,int u,int rk)
{
if(!u)return -1;
if(l==r)return l;
int mid = (l+r)>>1;
if(rk<=siz[ls[u]])return query(l,mid,ls[u],rk);
else return query(mid+1,r,rs[u],rk-siz[ls[u]]);
}
ll query(int l,int r,int u,int ql,int qr)
{
if(!u||ql>qr)return 0;
if(l==ql&&r==qr)return sum[u];
int mid = (l+r)>>1;
if(qr<=mid)return query(l,mid,ls[u],ql,qr);
else if(ql>mid)return query(mid+1,r,rs[u],ql,qr);
else return query(l,mid,ls[u],ql,mid)+query(mid+1,r,rs[u],mid+1,qr);
}
ll Query(int l,int r,int u,int ql,int qr)
{
if(!u||ql>qr)return 0;
if(l==ql&&r==qr)return siz[u];
int mid = (l+r)>>1;
if(qr<=mid)return Query(l,mid,ls[u],ql,qr);
else if(ql>mid)return Query(mid+1,r,rs[u],ql,qr);
else return Query(l,mid,ls[u],ql,mid)+Query(mid+1,r,rs[u],mid+1,qr);
}
ll query(int siz)
{
int mid = query(0,inf,rt,siz);
ll ql = query(0,inf,rt,0,mid-1),qr = query(0,inf,rt,mid+1,inf),tmp = (Query(0,inf,rt,mid+1,inf)-Query(0,inf,rt,0,mid-1));
return qr-ql-tmp*mid;
}
}tr[2];
int main()
{
// freopen("tt.in","r",stdin);
read(K),read(n);
for(int x,y,i=1;i<=n;i++)
{
scanf("%s",op1),read(x),scanf("%s",op2),read(y);
if(x>y)swap(x,y);
if(op1[0]==op2[0])
{
ans+=y-x;
i--,n--;
}else p[i]=Point(x,y);
}
ans+=n;
if(!n)
{
printf("%lld\n",ans);
return 0;
}
if(K==1)
{
for(int i=1;i<=n;i++)
s[++tl]=p[i].l,s[++tl]=p[i].r;
sort(s+1,s+tl+1);
int mid = (s[tl/2]+s[tl/2+1])/2;
int i;
for(i=1;s[i]<=mid;i++)ans+=mid-s[i];
for(;i<=tl;i++)ans+=s[i]-mid;
printf("%lld\n",ans);
return 0;
}
sort(p+1,p+1+n);
for(int i=1;i<=n;i++)
tr[1].insert(0,inf,tr[1].rt,p[i].l),tr[1].insert(0,inf,tr[1].rt,p[i].r);
ll mx = 0x3f3f3f3f3f3f3f3fll;
for(int i=1;i<=n;i++)
{
tr[0].insert(0,inf,tr[0].rt,p[i].l),tr[0].insert(0,inf,tr[0].rt,p[i].r);
tr[1].erase (0,inf,tr[1].rt,p[i].l),tr[1].erase (0,inf,tr[1].rt,p[i].r);
mx=min(mx,tr[0].query(i)+tr[1].query(n-i));
}
printf("%lld\n",ans+mx);
return 0;
}
APIO2015 八邻旁之桥/巴邻旁之桥的更多相关文章
- 洛谷 P3644 [APIO2015]八邻旁之桥 解题报告
P3644 [APIO2015]八邻旁之桥 题目描述 一条东西走向的穆西河将巴邻旁市一分为二,分割成了区域\(A\)和区域\(B\). 每一块区域沿着河岸都建了恰好\(1000000001\)栋的建筑 ...
- 【BZOJ4071】[Apio2015]巴邻旁之桥 Treap
[BZOJ4071][Apio2015]巴邻旁之桥 Description 一条东西走向的穆西河将巴邻旁市一分为二,分割成了区域 A 和区域 B. 每一块区域沿着河岸都建了恰好 1000000001 ...
- [APIO2015]八邻旁之桥——非旋转treap
题目链接: [APIO2015]八邻旁之桥 对于$k=1$的情况: 对于起点和终点在同侧的直接计入答案:对于不在同侧的,可以发现答案就是所有点坐标与桥坐标的差之和+起点与终点不在同一侧的人数. 将所有 ...
- 4071: [Apio2015]巴邻旁之桥
Description 一条东西走向的穆西河将巴邻旁市一分为二,分割成了区域 A 和区域 B. 每一块区域沿着河岸都建了恰好 1000000001 栋的建筑,每条岸边的建筑都从 0 编号到 10000 ...
- [BZOJ4071][APIO2015]八邻旁之桥
BZOJ(这题是BZOJ权限题,有权限号的就去看看吧) Luogu(良心洛谷) 题目描述 一条东西走向的穆西河将巴邻旁市一分为二,分割成了区域\(A\)和区域\(B\). 每一块区域沿着河岸都建了恰好 ...
- BZOJ4071 & 洛谷3644 & UOJ112:[APIO2015]巴邻旁之桥——题解
https://www.lydsy.com/JudgeOnline/problem.php?id=4071 https://www.luogu.org/problemnew/show/P3644 ht ...
- [bzoj4071] [Apio2015]巴邻旁之桥
Description 一条东西走向的穆西河将巴邻旁市一分为二,分割成了区域 A 和区域 B. 每一块区域沿着河岸都建了恰好 1000000001 栋的建筑,每条岸边的建筑都从 0 编号到 10000 ...
- 【BZOJ4071】【APIO2015】巴邻旁之桥
题意: Description 一条东西走向的穆西河将巴邻旁市一分为二,分割成了区域 A 和区域 B. 每一块区域沿着河岸都建了恰好 1000000001 栋的建筑,每条岸边的建筑都从 0 编号到 1 ...
- Appium+python自动化(二十八)- 滑呀滑,滑到奈何桥喝碗孟婆汤 - 高级滑动(超详解)
简介 奈何桥上叹奈何,三生石前憾三生,彼岸花下非彼岸,奈何三生彼岸人. 相传过了鬼门关便上一条路叫黄泉路,路上盛开着只见花,不见叶的彼岸花.花叶生生两不见,相念相惜永相失,路尽头有一条河叫忘川河,河上 ...
随机推荐
- 浅谈归并排序:合并 K 个升序链表的归并解法
在面试中遇到了这道题:如何实现多个升序链表的合并.这是 LeetCode 上的一道原题,题目具体如下: 用归并实现合并 K 个升序链表 LeetCode 23. 合并K个升序链表 给你一个链表数组,每 ...
- 《操作系统导论》第14章 | 内存操作API
内存类型 在运行一个C程序的时候,会分配两种类型的内存.第一种称为栈内存,它的申请和释放操作是编译器来隐式管理的,所以有时也称为自动内存.假设需要在func()函数中为一个整形变量x申请空间,我们只需 ...
- java中的运算符介绍
运算符&和&&的区别&运算符有两种用法:(1)按位与:(2)逻辑与. &&运算符是短路与运算.逻辑与跟短路与的差别是非常巨大的,虽然二者都要求运算符左右 ...
- 一个含有多个flag的图片(Misc)
图片是来自一个老阿姨,然后这个图片是属于一个杂项题目,一个图片中包含十几个flag,格式为#....#,第一个flag就是图片一开始就放在上面的,可以直接看到. 然后文件名字也是一个flag, 将图片 ...
- msf生成后门实战漫游内网
前言:当我们在渗透当中首先拿到webshell权限,进一步提权渗透,获得内网的主机,假设在这里我们获取了具有双网卡的web服务器,我们的目的是通过这台web服务器,进行内网漫游,获取内网中其他主机的系 ...
- [数分笔记]用Dedekind切割定理证明确界定理
1.定理内容 Dedekind切割定理:设是实数集的一个切割,则或者有最大数,或者有最小数. 确界定理:非空有上界的数集必有上确界,非空有下界的数集必有下确界. 2.证明过程 设非空数集有上界 记,即 ...
- 李宏毅机器学习笔记——新手感悟——PM2.5作业
python也不会,计算机也不会,啥都不会,只带了个脑子考了计算机研究生.研究生选了人工智能方向.看来注定是漫长的学习之旅. PM2.5作业,我是一个字都看不懂.所以我采用了直接看答案的方案.把答案看 ...
- 现在的BI软件是不是很贵?
目前一个企业光有现在狭义的拖拉拽自助 BI 够用吗?那明显是不够的!那么企业应该需要什么样的BI系统? 一个很多企业真正需要的 BI 解决方案一般有一下几类: 1.数据呈现 这是最关键也最基本的功能, ...
- 【c#新手学习 练习 案例】 阶段项目一:开发团队调度软件
案例是模仿java https://blog.csdn.net/bjfu170203101/article/details/109322590 改用C#:开发环境 vs2022/vscode .n ...
- SQL Server Cross/Outer Apply
SQL Server2005引入了APPLY运算符,它非常像连接子句,它允许两个表达式直接进行连接,即将左/外部表达式和右/内部表达式连接起来. CROSS APPLY(类比inner join)和O ...