hdu 3642(线段树+扫描线)

Posted wangwanxiang

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了hdu 3642(线段树+扫描线)相关的知识,希望对你有一定的参考价值。

三维扫描线,枚举z寻找相交区间的立方体,然后直接扫描线求xy平面的相交三次及以上面积,乘以z区间求和就可以了

#include <iostream>
#include <cstring>
#include <algorithm>
#include <cstdio>
using namespace std;
const int maxn=1e6+100;
const int maxm=2000+500;
int col[maxn<<2];
int len1[maxn<<2],len2[maxn<<2],sum[maxn<<2];//重合线段分别为1 2 3 的线段长度
int zz[maxm],xx[maxm];
typedef long long ll;
int t,n;
struct Point
{
    int x,y,z;
    void read()
    {
        scanf("%d%d%d",&x,&y,&z);
    }
};
struct graphic
{
    Point a,b;
}gg[maxm];
struct note
{
    int l,r,h;
    int z;
    int ff;
    note() {}
    note(int l,int r,int h,int ff):l(l),r(r),h(h),ff(ff) {}
    bool operator <(const note &p) const
    {
        return h<p.h;
    }
}aa[maxm];
void pushup(int l,int r,int rt)
{
    if(col[rt]>=3)
    {
        sum[rt]=xx[r+1]-xx[l];
        len1[rt]=len2[rt]=0;
    }
    else if(col[rt]==2)
    {
        sum[rt]=sum[rt<<1]+sum[rt<<1|1]+len1[rt<<1]+len1[rt<<1|1]+len2[rt<<1]+len2[rt<<1|1];
        len2[rt]=xx[r+1]-xx[l]-sum[rt];//总长度减去重合次数>=3的部分才是重合次数==2的部分
        len1[rt]=0;
    }
    else if(col[rt]==1)
    {
        sum[rt]=sum[rt<<1]+sum[rt<<1|1]+len2[rt<<1]+len2[rt<<1|1];
        len2[rt]=len1[rt<<1]+len1[rt<<1|1];
        len1[rt]=xx[r+1]-xx[l]-sum[rt]-len2[rt];//总长度减去重合次数>=和==2的部分才是重合次数==1的部分
    }
    else if(l==r)
    {
        sum[rt]=len1[rt]=len2[rt]=0;
    }
    else
    {
        sum[rt]=sum[rt<<1]+sum[rt<<1|1];
        len1[rt]=len1[rt<<1]+len1[rt<<1|1];
        len2[rt]=len2[rt<<1]+len2[rt<<1|1];
    }
}
void update(int L,int R,int v,int l,int r,int rt)
{
    if(L<=l&&r<=R)
    {
        col[rt]+=v;
        pushup(l,r,rt);
        return;
    }
    int mid=(l+r)>>1;
    if(L<=mid) update(L,R,v,l,mid,rt<<1);
    if(mid<R) update(L,R,v,mid+1,r,rt<<1|1);
    pushup(l,r,rt);
}
int main()
{
    scanf("%d",&t);
    int Case=0;
    while(t--)
    {
        scanf("%d",&n);
        int cx=0,cz=0;
         for(int i=1;i<=n;i++)
         {
             gg[i].a.read();
             gg[i].b.read();
             zz[cz++]=gg[i].a.z;
             zz[cz++]=gg[i].b.z;
             xx[cx++]=gg[i].a.x;
             xx[cx++]=gg[i].b.x;
         }
         printf("Case %d: ",++Case);
         if(n<3)
         {
             printf("0
");
             continue;
         }
         sort(zz,zz+cz);
         sort(xx,xx+cx);
         cz=unique(zz,zz+cz)-zz;
         cx=unique(xx,xx+cx)-xx;
         int nn=2000+10;
         ll ans=0;
         for(int j=0;j<cz-1;j++)//离散化z后枚举区间,寻找各个立方体和区间重合的部分
         {
             int cnt=0;
             for(int i=1;i<=n;i++)
             {
                 if(gg[i].a.z<=zz[j]&&gg[i].b.z>zz[j])//枚举的区间是离散化z后的,所以必然小于任意一个立方体的高
                 {
                     aa[++cnt]=note(gg[i].a.x,gg[i].b.x,gg[i].a.y,1);
                     aa[++cnt]=note(gg[i].a.x,gg[i].b.x,gg[i].b.y,-1);
                 }
             }
             sort(aa+1,aa+cnt+1);
             ll sss=0;
             for(int i=1;i<=cnt;i++)
             {
                 int hl=lower_bound(xx,xx+cx,aa[i].l)-xx;
                 int hr=lower_bound(xx,xx+cx,aa[i].r)-xx;
                 sss+=(ll)sum[1]*(ll)(aa[i].h-aa[i-1].h);
                 update(hl,hr-1,aa[i].ff,0,nn,1);
             }
             ans+=sss*(ll)(zz[j+1]-zz[j]);//重合三次后的面积乘以高
         }
         printf("%lld
",ans);
    }
    return 0;
}

 

以上是关于hdu 3642(线段树+扫描线)的主要内容,如果未能解决你的问题,请参考以下文章

HDU3642 Get The Treasury —— 求矩形交体积 线段树 + 扫描线 + 离散化

HDU-3642 Get The Treasury(扫描线 + 离散化 + 线段树)

HDU 3642 Get The Treasury (线段树扫描线)

hdu 3642 Get The Treasure

hdu1542 Atlantis(扫描线+线段树+离散)矩形相交面积

hdu 1542 线段树之扫描线之面积并