题目描述

近来A国和B国的矛盾激化,为了预防不测,A国准备修建一条长长的防线,当然修建防线的话,肯定要把需要保护的城市修在防线内部了。可是A国上层现在还犹豫不决,到底该把哪些城市作为保护对象呢?又由于A国的经费有限,所以希望你能帮忙完成如下的一个任务:

  1. 给出你所有的A国城市坐标

  2. A国上层经过讨论,考虑到经济问题,决定取消对i城市的保护,也就是说i城市不需要在防线内了

  3. A国上层询问对于剩下要保护的城市,修建防线的总经费最少是多少

你需要对每次询问作出回答。注意单位1长度的防线花费为1。

A国的地形是这样的,形如下图,x轴是一条河流,相当于一条天然防线,不需要你再修建

A国总是有两个城市在河边,一个点是(0,0),一个点是(n,0),其余所有点的横坐标均大于0小于n,纵坐标均大于0。A国有一个不在(0,0)和(n,0)的首都。(0,0),(n,0)和首都这三个城市是一定需要保护的。

说明

数据范围:

30%的数据m<=1000,q<=1000

100%的数据m<=100000,q<=200000,n>1

所有点的坐标范围均在10000以内, 数据保证没有重点

题解

要动态维护一个凸包。还要维护删除操作?

删除麻烦,可以离线,把删除变成插入操作。

一切就简单又自然了。

插入一个点t,就要找到凸包上的第一个大于等于横坐标x的点p,和第一个小于x的点q。即后继前驱。

题目很善良,河边点不会删除,所以一定在凸包上,不会有什么边界的锅。

如果在凸包里面那么就直接返回,怎么判断?

如果q和p的斜率在t和p的斜率,t和q的斜率大小之间的话,那么这个点一定不在凸包上。画图可以理解。

反之就一定在凸包上。

然后开始弹出点。不断向右比较斜率,再不断向左比较斜率。

最后把t加入凸包。

删除点的时候,实时更新防线的长度。

不要忘了最后把p、q之间的连边删除。

找前驱后继再删除,可以用splay实现。

但是没有必要,set完全可以支持。

代码:

#include<bits/stdc++.h>
using namespace std;
const int N=+;
const double inf=19260817.0;
int n,m;
int cx,cy,far;
double now;
struct que{
int typ;
int id;
double ans;
}q[*N];
struct po{
int x,y;
bool friend operator<(po a,po b){
return a.x<b.x;
}
void op(){
cout<<" point "<<x<<" "<<y<<endl;
}
}a[N];
bool die[N];
set<po>s;
set<po>::iterator it1,it2,it3;
double dis(po a,po b){
return sqrt((double)(a.x-b.x)*(a.x-b.x)+(double)(a.y-b.y)*(a.y-b.y));
}
double slo(po a,po b){//(x1,y1) -> (x2,y2)
if(a.x==b.x){
return a.y>b.y?-inf:inf;
}
return ((double)a.y-(double)b.y)/((double)a.x-(double)b.x);
}
void upda(po lp){
//lp.op();
//cout<<" size "<<s.size()<<endl;
it2=s.lower_bound(lp);
it1=it2;it1--;
if(slo(*it1,lp)<=slo(*it1,*it2)&&slo(*it1,*it2)<=slo(lp,*it2)){
return;//has been protected;
}
now-=dis(*it1,*it2);
it3=it2;it3++;
while(it3!=s.end()&&slo(lp,*it2)<=slo(lp,*it3)){
now-=dis(*it2,*it3);
s.erase(it2);
it2=it3;
it3++;
}
now+=dis(lp,*it2);
it3=it1;it3--;
while(it1!=s.begin()&&slo(lp,*it1)>=slo(lp,*it3)){
now-=dis(*it1,*it3);
s.erase(it1);
it1=it3;
it3--;
}
now+=dis(lp,*it1);
s.insert(lp);
}
int main()
{
scanf("%d%d%d",&far,&cx,&cy);
scanf("%d",&n);int x,y;
for(int i=;i<=n;i++){
scanf("%d%d",&x,&y);
a[i].x=x,a[i].y=y;
}
scanf("%d",&m);
for(int i=;i<=m;i++){
scanf("%d",&q[i].typ);
if(q[i].typ==) scanf("%d",&q[i].id),die[q[i].id]=;
}
po st;st.x=,st.y=;s.insert(st);
po nd;nd.x=far,nd.y=;s.insert(nd);
po ca;ca.x=cx,ca.y=cy;s.insert(ca);
now=dis(st,ca)+dis(ca,nd);
for(int i=;i<=n;i++){
if(!die[i]){
upda(a[i]);
}
}
for(int i=m;i>=;i--){
if(q[i].typ==){
q[i].ans=now;
}
else{
upda(a[q[i].id]);
}
}
for(int i=;i<=m;i++){
if(q[i].typ==){
printf("%.2lf\n",q[i].ans);
}
}
return ;
}

总结:

现在我们有了一些斜率优化中,维护凸包的方法。

1.单调队列,适用于斜率有单调性,并且x要有单调性。才可以直接队头弹出,队尾插入。均摊O(1)

2.队列+二分。对于加入点的x有单调性,但是查询的斜率无单调性的时候,就要二分了。

二分到第一个斜率大于/小于查询斜率的点,作为决策点即可。除了队尾加入的时候,为了维护凸包所需,不会从队头弹出点。】

3.set(平衡树)维护。对于一般情况的凸包,可能加入的x在任何位置,查询什么凸包的周长,就必须用set了。

但是,对于斜率优化中,要查询一个第一个和后继/前驱比k大/小的点,因为set是按照x重载的运算符,不能直接lower_bound了。

所以,只能手写一棵splay,(就我所知)别无他法。

具体来说,一个splay树上的节点,按照x排序,而且必须还要记录和后继的斜率slope,和子树内所有后继斜率slope的最小值,便于直接二分。

[HAOI2011]防线修建的更多相关文章

  1. 【BZOJ 2300】 2300: [HAOI2011]防线修建 (动态凸包+set)

    2300: [HAOI2011]防线修建 Description 近来A国和B国的矛盾激化,为了预防不测,A国准备修建一条长长的防线,当然修建防线的话,肯定要把需要保护的城市修在防线内部了.可是A国上 ...

  2. BZOJ 2300: [HAOI2011]防线修建( 动态凸包 )

    离线然后倒着做就变成了支持加点的动态凸包...用平衡树维护上凸壳...时间复杂度O(NlogN) --------------------------------------------------- ...

  3. [luogu P2521] [HAOI2011]防线修建

    [luogu P2521] [HAOI2011]防线修建 题目描述 近来A国和B国的矛盾激化,为了预防不测,A国准备修建一条长长的防线,当然修建防线的话,肯定要把需要保护的城市修在防线内部了.可是A国 ...

  4. P2521 [HAOI2011]防线修建

    题目链接:P2521 [HAOI2011]防线修建 题意:给定点集 每次有两种操作: 1. 删除一个点 (除开(0, 0), (n, 0), 与指定首都(x, y)) 2. 询问上凸包长度 至于为什么 ...

  5. bzoj千题计划236:bzoj2300: [HAOI2011]防线修建

    http://www.lydsy.com/JudgeOnline/problem.php?id=2300 维护动态凸包,人懒用的set 用叉积判断,不要用斜率 #include<set> ...

  6. 【BZOJ2300】[HAOI2011]防线修建 set维护凸包

    [BZOJ2300][HAOI2011]防线修建 Description 近来A国和B国的矛盾激化,为了预防不测,A国准备修建一条长长的防线,当然修建防线的话,肯定要把需要保护的城市修在防线内部了.可 ...

  7. 【题解】P2521 [HAOI2011]防线修建(动态凸包)

    [题解]P2521 [HAOI2011]防线修建(动态凸包) 凸包是易插入不好删除的东西,按照剧情所以我们时光倒流 然后问题就是维护凸包的周长,支持加入 本来很简单,但是计算几何就是一些小地方经验不足 ...

  8. BZOJ2300[HAOI2011]防线修建——非旋转treap+凸包(平衡树动态维护凸包)

    题目描述 近来A国和B国的矛盾激化,为了预防不测,A国准备修建一条长长的防线,当然修建防线的话,肯定要把需要保护的城市修在防线内部了.可是A国上层现在还犹豫不决,到底该把哪些城市作为保护对象呢?又由于 ...

  9. LG2521 [HAOI2011]防线修建

    题意 题目描述 近来A国和B国的矛盾激化,为了预防不测,A国准备修建一条长长的防线,当然修建防线的话,肯定要把需要保护的城市修在防线内部了.可是A国上层现在还犹豫不决,到底该把哪些城市作为保护对象呢? ...

随机推荐

  1. AssertionError

    (1)p1 = multiprocessing.Process(test1)p2 = multiprocessing.Process(target=test2) 错误: p1缺少target,应为(t ...

  2. Codeforces70 | Codeforces Beta Round #64 | 瞎讲报告

    目录 前言 正文 A B C D E 前言 这个毒瘤的517 放了Div1 然后D题是昨天讲的动态凸包(啊喂!我还没来的及去写 结果自己想的是二分凸包 (当然没有写出来 写完前两题之后就愉快地弃疗 C ...

  3. Python基础_异常处理与跟踪

    异常的种类 AttributeError 试图访问一个对象没有的树形,比如foo.x,但是foo没有属性x IOError 输入/输出异常:基本上是无法打开文件 ImportError 无法引入模块或 ...

  4. DataTime日期格式化

    C# DateTime日期格式化 在C#中DateTime是一个包含日期.时间的类型,此类型通过ToString()转换为字符串时,可根据传入给Tostring()的参数转换为多种字符串格式. 目录 ...

  5. Java锁的种类以及辨析

    锁作为并发共享数据,保证一致性的工具,在JAVA平台有多种实现(如 synchronized 和 ReentrantLock等等 ) .这些已经写好提供的锁为我们开发提供了便利,但是锁的具体性质以及类 ...

  6. NetFPGA-SUME下reference_nic测试

    Reference_nic Reference_nic是NetFPGA-SUME中提供的一个参考Demo,本文主要介绍如何构建并在SUME上运行reference_nic. GIT源 git clon ...

  7. POJ 1185 炮兵阵地 状压dp

    题目链接: http://poj.org/problem?id=1185 炮兵阵地 Time Limit: 2000MS Memory Limit: 65536K 问题描述 司令部的将军们打算在N*M ...

  8. 第一次spring冲刺第9天

    明天是这个阶段的最后一天了,今天讨论关于容错的方面,例如输入空白或其他字符等方面会出现的问题 ,部分代码如下: public void checkout(int trueResult) { Strin ...

  9. 微服务负载均衡 —— ribbon

    负载均衡是系统高可用.缓解网络流量和处理能力扩容的重要手段,广义的负载均衡指的是服务端负载均衡,如硬件负载均衡(F5)和软件负载均衡(Nginx).负载均衡设备会维护一份可用的服务器的信息,当客户端请 ...

  10. Robot Framework 教程 (7) - 使用For循环

    在自动化测试过程中,使用For循环来对某个动作进行重复操作是很普遍的行为.在Robot Framework中,各种测试库中均提供了多种方式的For循环结构,在其中覆盖了大部分类型的循环类型.而Robo ...