zoukankan      html  css  js  c++  java
  • 2017 ACM/ICPC 南宁区 网络赛 Overlapping Rectangles

    2017-09-24 20:11:21

    writer:pprp

    找到的大神的代码,直接过了

    采用了扫描线+线段树的算法,先码了,作为模板也不错啊

    题目链接:https://nanti.jisuanke.com/t/17313

    题意:给你很多个矩形,让你得到矩形的面积,重叠部分只算一次

    代码如下:

    //ac F
    #include <stdio.h>
    #include <string.h>
    #include <stdlib.h>
    #include <iostream>
    #include <algorithm>
    
    using namespace std;
    typedef long long ll;
    const int maxn=1e6+10;
    
    #define mm(a) memset(a,0,sizeof(a))
    
    int num1[maxn*4];
    int num[maxn*4],X[maxn*4];
    struct edge
    {
        int l,r,h;
        int s;//s为1是下边,s为-1是上边
        edge() {};
        edge(int a,int b,int c,int d) : l(a),r(b),h(c),s(d) {}
        bool operator<(const edge &n)const
        {
            return h<n.h;
        }
    } ss[maxn];
    
    void pushup(int le,int ri,int node)
    {
        if(num1[node])
            num[node]=X[ri+1]-X[le];//在更新的时候,可能两个矩阵有重叠,这样就不能像以前那么更新,而是将le和ri传入
        else if(le==ri)
            num[node]=0;           //然后将X[ri+1]-X[le]的值进行更新,避免了重复的长度
        else
            num[node]=num[node<<1]+num[node<<1|1];
    }
    void update(int l,int r,int add,int le,int ri,int node)
    {
        if(l<=le&&ri<=r)
        {
            num1[node]+=add;//与懒惰标记类似
            pushup(le,ri,node);
            return ;
        }
        int t=(le+ri)>>1;
        if(l<=t) update(l,r,add,le,t,node<<1);
        if(r>t) update(l,r,add,t+1,ri,node<<1|1);
        pushup(le,ri,node);
    }
    int main()
    {
        int n;
        while(scanf("%d",&n) != EOF)
        {
            if(n == 0)
            {
                cout << "*" << endl;
                break;
            }
            int a,b,c,d;
            int k=0;
            for(int i=0; i<n; i++)
            {
                scanf("%d%d%d%d",&a,&b,&c,&d);
                X[k]=a;
                ss[k++]=edge(a,c,b,1);
                X[k]=c;
                ss[k++]=edge(a,c,d,-1);
            }
            sort(X,X+k);
            sort(ss,ss+k);
            int k1=1;
            for(int i=1; i<k; i++) //对X进行离散化
            {
                if(X[i]!=X[i-1]) X[k1++]=X[i];
            }
            mm(num);
            mm(num1);
            int ans=0;
            for(int i=0; i<k-1; i++)
            {
                int l=lower_bound(X,X+k1,ss[i].l)-X;
                int r=lower_bound(X,X+k1,ss[i].r)-X-1;
                update(l,r,ss[i].s,0,k1-1,1);
                ans+=num[1]*(ss[i+1].h-ss[i].h);//num[1]为当前横坐标的总长度
            }
            cout << ans << endl;
        }
        return 0;
    }
  • 相关阅读:
    flex布局知识总结
    js,ts操作dom总结
    编译原理 语法树 句柄 简单短语 短语
    linux基础命令期末考试总结
    arm汇编指令--str ldr
    npm常用命令(原创)
    JS获取图片的缩略图
    Spring MVC 返回Json IE出现下载
    jquery获取页面iframe内容
    MySQL 下 ROW_NUMBER / DENSE_RANK / RANK 的实现
  • 原文地址:https://www.cnblogs.com/pprp/p/7588383.html
Copyright © 2011-2022 走看看