题目大意:

http://www.lydsy.com/JudgeOnline/problem.php?id=2300

题解

这道题让我们维护一个支持动态删除点的上凸壳

并且告诉了我们三个一定不会被删除的点

也就是说,无论什么时候这都是一个凸包

如果我们考虑从凸包里删除一个点吗...这个点在凸包里还好说,不用管了

在凸包上嘛...哪位dalao写出来了请教教我,并不会删除QAQ

所以我们倒序考虑所有的操作,将其变成动态插入点

每一次插入一个点的时候,我们找到凸包上与这个点关联的两个点

(按照横纵坐标进行双关键字排序后与之相邻的两个点)

我们就从这两个点开始依次向左右拓展,并且即时更新凸包的长度

所以我们用set存一下所有在凸包上的点即可

#include <set>
#include <cmath>
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long ll;
inline void read(int &x){
x=0;char ch;bool flag = false;
while(ch=getchar(),ch<'!');if(ch == '-') ch=getchar(),flag = true;
while(x=10*x+ch-'0',ch=getchar(),ch>'!');if(flag) x=-x;
}
inline int cat_max(const int &a,const int &b){return a>b ? a:b;}
inline int cat_min(const int &a,const int &b){return a<b ? a:b;}
const int maxn = 100010;
const int maxm = 200010;
struct Point{
int x,y;
Point(const int &a = 0,const int &b = 0){x=a;y=b;}
void print(){
printf("Point : (%d,%d)\n",x,y);
}
};
typedef Point Vector;
Vector operator + (const Vector &a,const Vector &b){
return Vector(a.x+b.x,a.y+b.y);
}
Vector operator - (const Vector &a,const Vector &b){
return Vector(a.x-b.x,a.y-b.y);
}
bool operator < (const Point &a,const Point &b){
return a.x == b.x ? a.y < b.y : a.x < b.x;
}
int operator * (const Vector &a,const Vector &b){
return a.x*b.x + a.y*b.y;
}
inline int cross(const Vector &a,const Vector &b){
return a.x*b.y - a.y*b.x;
}
inline double sqr(const double &x){
return x*x;
}
inline double dis(Point a,Point b){
return sqrt(sqr(a.x - b.x) + sqr(a.y - b.y));
}
struct Node{
int cmd,x;
}quer[maxm];
set<Point>s;
Point p[maxn];bool vis[maxn];
double ans = .0;
double anss[maxm];int cnt;
void insert(Point x){
set<Point>::iterator r = s.lower_bound(x);
set<Point>::iterator l = --r;r++;
set<Point>::iterator it;
if(cross(x - *l,*r - *l) >= 0) return;
ans -= dis(*l,*r);
while(1){
it = r++;
if(r == s.end()) break;
if(cross(*r - *it,*it - x) > 0) break;
ans -= dis(*it,*r);s.erase(it);
}
ans += dis(x,*it);
while(1){
it = l--;
if(it == s.begin()) break;
if(cross(x - *it,*it - *l) > 0) break;
ans -= dis(*it,*l);s.erase(it);
}
ans += dis(x,*(it));
s.insert(x);
}
int main(){
int n,x,y;read(n);read(x);read(y);
s.insert(Point(0,0));
s.insert(Point(x,y));
s.insert(Point(n,0));
ans += dis(Point(0,0),Point(x,y)) + dis(Point(x,y),Point(n,0));
read(n);
for(int i=1;i<=n;++i){
read(p[i].x);
read(p[i].y);
}
int m;read(m);
for(int i=1;i<=m;++i){
read(quer[i].cmd);
if(quer[i].cmd == 1){
read(quer[i].x);
vis[quer[i].x] = true;
}
}
for(int i=1;i<=n;++i){
if(!vis[i]) insert(p[i]);
}
for(int i=m;i>=1;--i){
if(quer[i].cmd == 1) insert(p[quer[i].x]);
else anss[++cnt] = ans;
}
for(int i=cnt;i>=1;--i) printf("%.2lf\n",anss[i]);
getchar();getchar();
return 0;
}

bzoj 2300: [HAOI2011]防线修建 凸包的更多相关文章

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

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

  2. bzoj 2300 [HAOI2011]防线修建 set动态维护凸包

    题目大意 动态删点,求凸包周长 分析 反过来变成动态加点 用set维护平衡树 具体是找到凸包上左右两点 拆开 就可以把左边当作顺时针求的一个凸包,右边当作逆时针求的一个凸包,像栈那样出set就好了 注 ...

  3. bzoj 2300 : [HAOI2011]防线修建

    set动态维护凸包 #include<iostream> #include<cstdio> #include<cstring> #include<algori ...

  4. BZOJ 2300 [HAOI2011]防线修建 ——计算几何

    只需要倒着插入,然后维护一个凸包就可以了. 可以用来学习set的用法 #include <map> #include <set> #include <cmath> ...

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

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

  6. bzoj2300#2300. [HAOI2011]防线修建

    题解:带删点的维护凸包,1.删点2.查询凸包周长 题解:倒着做就成了带加点的维护凸包,加点时维护一下周长就没了 //#pragma GCC optimize(2) //#pragma GCC opti ...

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

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

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

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

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

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

随机推荐

  1. RabbitMQ与Redis做队列比较

    本文仅针对RabbitMQ与Redis做队列应用时的情况进行对比 具体采用什么方式实现,还需要取决于系统的实际需求简要介绍RabbitMQRabbitMQ是实现AMQP(高级消息队列协议)的消息中间件 ...

  2. redis 集群 搭建

    环境: centos6.5 192.168.16.11 centos6.5 192.168.16.12 centos6.5 192.168.16.13 三台虚拟机模拟9个节点,一台机器3个节点,创建出 ...

  3. 【BZOJ2466】[中山市选2009]树 树形DP

    [BZOJ2466][中山市选2009]树 Description 图论中的树为一个无环的无向图.给定一棵树,每个节点有一盏指示灯和一个按钮.如果节点的按扭被按了,那么该节点的灯会从熄灭变为点亮(当按 ...

  4. swift打印方法名文件路径

    1.打印方法名 print(#function) 1.打印文件路径 print(#file)

  5. 我的Android进阶之旅------>对Android开发者有益的40条优化建议

    下面是开始Android编程的好方法: 找一些与你想做事情类似的代码 调整它,尝试让它做你像做的事情 经历问题 使用StackOverflow解决问题 对每个你像添加的特征重复上述过程.这种方法能够激 ...

  6. 超轻量级、高性能C日志库--EasyLogger

    [ 声明:版权全部,欢迎转载.请勿用于商业用途. 联系信箱:armink.ztl@gmail.com] EasyLogger 1. 介绍 EasyLogger 是一款超轻量级(ROM<1.6K, ...

  7. 【Android】开源项目汇总

    Android开源项目第一篇——个性化控件(View)篇  包括ListView.ActionBar.Menu.ViewPager.Gallery.GridView.ImageView.Progres ...

  8. ABAP下载xml文件

    [转http://www.cnblogs.com/byfhd/archive/2007/08/17/859829.html] ************************************* ...

  9. 数据库存储I/O类型分析与配置

    存储设备作为数据的容器,为应用提供数据存取服务,而存储系统将数据展现给不同的应用后,应用程序对数据访问不尽相同.简要来说,就是读和写,更加细分的话是以不同的传输单元(I/O大小)进行顺序和随机类型的读 ...

  10. Yii2 关于电子商务的开源项目

    https://github.com/samdark/yii2-shop https://github.com/omnilight/yii2-shopping-cart https://github. ...