zoukankan      html  css  js  c++  java
  • bzoj 2300 [HAOI2011]防线修建 set动态维护凸包

    题目大意

    动态删点,求凸包周长

    分析

    反过来变成动态加点
    用set维护平衡树
    具体是找到凸包上左右两点
    拆开
    就可以把左边当作顺时针求的一个凸包,右边当作逆时针求的一个凸包,像栈那样出set就好了
    注意新点在凸包内不用管它
    每个点进一次出一次
    (O(n log n))

    solution

    #include <cstdio>
    #include <cstdlib>
    #include <cstring>
    #include <cmath>
    #include <cctype>
    #include <algorithm>
    #include <set>
    using namespace std;
    typedef double db;
    const db eps=1e-7;
    const int M=200007;
    
    inline int rd(){
    	int x=0;bool f=1;char c=getchar();
    	for(;!isdigit(c);c=getchar()) if(c=='-') f=0;
    	for(;isdigit(c);c=getchar()) x=x*10+c-48;
    	return f?x:-x;
    }
    
    db sum;
    int n,m;
    int kd[M],opr[M];
    bool vis[M];
    db ans[M];
    
    struct pt{
    	db x,y;
    	pt(db _x=0.0,db _y=0.0){x=_x; y=_y;}
    }p[M];
    /*精度已经把我搞啥了我都不知道加不加,加了set直接炸
    bool eq(db x,db y){return fabs(x-y)<=eps;}
    bool neq(db x,db y){return !eq(x,y);}
    bool le(db x,db y){return eq(x,y)||x<y;}
    bool ge(db x,db y){return eq(x,y)||x>y;}
    */
    pt operator -(pt x,pt y){return pt(x.x-y.x,x.y-y.y);}
    pt operator +(pt x,pt y){return pt(x.x+y.x,x.y+y.y);}
    pt operator *(pt x,db d){return pt(x.x*d,x.y*d);}
    pt operator /(pt x,db d){return pt(x.x/d,x.y/d);}
    bool operator <(pt x,pt y){if(x.x!=y.x)return x.x<y.x;return x.y<y.y;}
    
    db dot(pt x,pt y){
    	return x.x*y.x+x.y*y.y;
    }
    
    db cross(pt x,pt y){
    	return x.x*y.y-x.y*y.x;
    }
    
    db area(pt x,pt y,pt z){
    	return cross(y-x,z-x);
    }
    
    db length(pt x){
    	return sqrt(dot(x,x));
    }
    
    set<pt>S;
    typedef set<pt>::iterator its;
    
    void gao(pt x){
    	its r=S.lower_bound(x);
    	its l=r; --l;
    	if(area(*l,x,*r)>=0) return;//凸包内
    	sum-=length(*r-*l);
    	its tp;
    	while(1){
    		tp=r;
    		++r;
    		if(r==S.end()) break;
    		if(area(*r,*tp,x)<=0){
    			sum-=length(*tp-*r);
    			S.erase(tp);
    		}
    	}
    	while(1){
    		if(l==S.begin()) break;
    		tp=l;
    		--l;
    		if(area(*l,*tp,x)>=0){
    			sum-=length(*tp-*l);
    			S.erase(tp);
    		}
    		else break;
    	}
    	S.insert(x);
    	l=r=S.find(x);
    	l--; r++;
    	sum+=length(x-*l);
    	sum+=length(x-*r);
    }
    
    int main(){
    	int i,z,x,y;
    	
    	z=rd(),x=rd(),y=rd();
    	
    	pt nw=pt(0,0);
    	S.insert(nw);
    	
    	nw=pt(z,0);
    	S.insert(nw);
    	
    	nw=pt(x,y);
    	S.insert(nw);
    	
    	sum=length(pt(x,y))+length(pt(z-x,y));
    	
    	n=rd();
    	for(i=1;i<=n;i++){
    		x=rd(),y=rd();
    		p[i]=pt(x,y);
    	}
    	
    	m=rd();
    	for(i=1;i<=m;i++){
    		kd[i]=rd();
    		if(kd[i]==1){
    			opr[i]=rd();
    			vis[opr[i]]=1;
    		}
    	}
    	
    	for(i=1;i<=n;i++)
    		if(!vis[i]) gao(p[i]);
    		
    	for(i=m;i>0;i--){
    		if(kd[i]==2) ans[i]=sum;
    		else gao(p[opr[i]]);
    	}
    	
    	for(i=1;i<=m;i++) if(kd[i]==2) printf("%.2lf
    ",ans[i]);
    	
    	return 0;
    }
    
  • 相关阅读:
    php多进程和多线程的比较
    设计模式学习系列——建造者模式
    设计模式学习系列——单例模式
    设计模式学习系列——前言
    设计模式学习系列——工厂模式
    记一次给nginx的web服务器目录加软链接
    某公司后端开发工程师面试题学习
    2010年腾讯前端面试题学习(jquery,html,css部分)
    2010年腾讯前端面试题学习(js部分)
    winfrom 隐藏任务栏(win7)
  • 原文地址:https://www.cnblogs.com/acha/p/6431758.html
Copyright © 2011-2022 走看看