hdu1542

链接:点击打开链接

题意 :给出n个矩形的左下角和右上角的坐标,求矩形面积的并

hdu1542_第1张图片hdu1542_第2张图片

代码:
#include <stdio.h>
#include <string.h>
#include <iostream>
#include <algorithm>
using namespace std;
const int SIZE=505;
int add[SIZE<<2];                               //add为区间标记,与懒惰标记类似
double x[SIZE<<2],sum[SIZE<<2];
struct node{
    int ss;                                     //ss=1为下边,ss=-1为上边
    double l,r,h;                               //分别记录线段的左端,右端和高度
    node(){}
    node(double a,double b,double c,int d):l(a),r(b),h(c),ss(d){}
    friend bool operator<(node p,node q){       //因为是从下到上扫描因此,需要按照高度排序
        return p.h<q.h;
    }
}s[SIZE];
void pushup(int rt,int l,int r){
    if(add[rt])                                 //这就是如何解决的线段重叠的办法,如果当前区间
    sum[rt]=x[r+1]-x[l];                        //有标记用x数组更新,而不是左右儿子更新
    else if(l==r)
    sum[rt]=0;
    else
    sum[rt]=sum[rt<<1]+sum[rt<<1|1];            //与懒惰标记的区别在于标记不会消失(除非遇到上边)
}                                               //这也就是线段不会重复相加减的关键
void update(int L,int R,int c,int l,int r,int rt){
    int m;
    if(L<=l&&r<=R){
        add[rt]+=c;                             //update与普通线段树基本相同
        pushup(rt,l,r);
        return;
    }
    m=(l+r)>>1;
    if(L<=m)
    update(L,R,c,l,m,rt<<1);
    if(R>m)
    update(L,R,c,m+1,r,rt<<1|1);
    pushup(rt,l,r);
}
int main(){                                     //用的是从下到上扫描,也就是扫描线,又因为
    int n,i,k,l,m,r,cas;                        //x坐标可能过多,因此需要对x轴进行离散化,从
    double a,b,c,d,ans;                         //下到上扫描也就是遇到下边进行成段加1,遇到
    cas=1;                                      //上边就是成段减1,所以问题就变成了线段树成
    while(scanf("%d",&n)!=EOF&&n){              //段更新,但我们还需要解决线段重叠的问题,只
        k=1,ans=m=0;                            //要修改pushup就能巧妙的解决这个问题
        for(i=0;i<n;i++){
            scanf("%lf%lf%lf%lf",&a,&b,&c,&d);
            x[m]=a;
            s[m++]=node(a,c,b,1);
            x[m]=c;
            s[m++]=node(a,c,d,-1);
        }
        sort(x,x+m);
        sort(s,s+m);
        for(i=1;i<m;i++)                        //对x坐标进行离散化,也可以用set
        if(x[i]!=x[i-1])
        x[k++]=x[i];
        memset(add,0,sizeof(add));
        memset(sum,0,sizeof(sum));
        for(i=0;i<m-1;i++){
            l=lower_bound(x,x+k,s[i].l)-x;
            r=lower_bound(x,x+k,s[i].r)-x-1;    //找出更新的区间,这是左必右开区间,因此
            update(l,r,s[i].ss,0,k-1,1);        //r需要减1,这也是与普通线段树的区别,普通
            ans+=(sum[1]*(s[i+1].h-s[i].h));    //的线段树其实是更新的节点中保存的点,假如
        }                                       //对[1,2]和[2,3]区间都加1,普通线段树中2
        printf("Test case #%d\nTotal explored area: %.2lf\n\n",cas++,ans);
    }                                           //那个点会变成2,而这个是表示的线段,因此
    return 0;                                   //2应该为1,因此用左闭右开区间
}

你可能感兴趣的:(hdu1542)