面积并+扫描线 覆盖的面积 HDU - 1255

题目链接:https://cn.vjudge.net/problem/HDU-1255

题目大意:中文题目

具体思路:和上一篇的博客思路差不多,上一个题求的是面积,然后我们这个地方求的是啊覆盖两次及两次以上的面积,我们可以在原来的基础上进行改进,原来的tree1储存的是覆盖一次的合理的面积,我们再加一个tree2求得是覆盖两次及以上的面积,具体的判断过程:

1,如果lazy[rt]>1,就代表这块区域完全的被覆盖了两次,那么这块区域的面积就是hash【r+1】-hash【l】。

2,如果是根节点,覆盖两次的面积是0,我们判断是l+1==r。

3,如果是lazy[rt]==1的话,我们就取原来覆盖一次的面积上已经被覆盖的,这样就相当于求的是覆盖两次的面积。

4,其余的话,就是两个根节点覆盖两次的面积了。

AC代码:

  1 #include <iostream>
  2 #include <cstdio>
  3 #include <string>
  4 #include <cstring>
  5 #include <algorithm>
  6 #include <cmath>
  7 #include <vector>
  8 #include <queue>
  9 #include <stack>
 10 #include <ctime>
 11 #define ll long long
 12 # define lson l,m,rt<<1
 13 # define rson m+1,r,rt<<1|1
 14 using namespace std;
 15 const int maxn = 1000+100;
 16 # define inf 0x3f3f3f3f
 17 struct node
 18 {
 19     double l,r,h;
 20     int d;
 21     node() {}
 22     node(double xx,double yy,double zz,int tt)
 23     {
 24         l=xx;
 25         r=yy;
 26         h=zz;
 27         d=tt;
 28     }
 29     bool friend operator < (node t1,node t2)
 30     {
 31         return t1.h<t2.h;
 32     }
 33 } q[maxn<<2];
 34 double Hash[maxn<<2],tree1[maxn<<2],tree2[maxn<<2],lazy[maxn<<2];
 35 void up(int l,int r,int rt)
 36 {
 37     if(lazy[rt])
 38         tree1[rt]=Hash[r+1]-Hash[l];
 39     else if(l==r)
 40         tree1[rt]=0;
 41     else
 42         tree1[rt]=tree1[rt<<1]+tree1[rt<<1|1];
 43     if(lazy[rt]>1)
 44         tree2[rt]=Hash[r+1]-Hash[l];
 45     else if(l==r)
 46         tree2[rt]=0;
 47         else if(lazy[rt]==1)
 48             tree2[rt]=tree1[rt<<1]+tree1[rt<<1|1];
 49     else
 50         tree2[rt]=tree2[rt<<1]+tree2[rt<<1|1];
 51 }
 52
 53 void update(int l,int r,int rt,int L,int R,int p)
 54 {
 55     if(L<=l&&R>=r)
 56     {
 57         lazy[rt]+=p;
 58         up(l,r,rt);
 59         return ;
 60     }
 61     int m=(l+r)>>1;
 62     if(L<=m)
 63         update(lson,L,R,p);
 64     if(R>m)
 65         update(rson,L,R,p);
 66     up(l,r,rt);
 67 }
 68 int main()
 69 {
 70     int T;
 71     scanf("%d",&T);
 72     while(T--)
 73     {
 74         int n,num=0;
 75         scanf("%d",&n);
 76         double x1,y1,x2,y2;
 77         for(int i=0; i<n; i++)
 78         {
 79              scanf("%lf %lf %lf %lf",&x1,&y1,&x2,&y2);
 80             q[num]= {x1,x2,y1,1};
 81             Hash[num++]=x1;
 82             q[num]= {x1,x2,y2,-1};
 83             Hash[num++]=x2;
 84         }
 85         sort(Hash,Hash+num);
 86         sort(q,q+num);
 87         int k=1;
 88         for(int i=1; i<num; i++)
 89         {
 90             if(Hash[i]==Hash[i-1])
 91                 continue;
 92             Hash[k++]=Hash[i];
 93         }
 94         memset(tree1,0,sizeof(tree1));
 95         memset(tree2,0,sizeof(tree2));
 96         memset(lazy,0,sizeof(lazy));
 97         double ans=0;
 98     //    cout<<1<<endl;
 99         for(int i=0; i<num; i++)
100         {
101         //    cout<<i<<endl;
102             int l=lower_bound(Hash,Hash+k,q[i].l)-Hash;
103             int r=lower_bound(Hash,Hash+k,q[i].r)-Hash-1;
104             update(0,k-1,1,l,r,q[i].d);
105             ans+=tree2[1]*(q[i+1].h-q[i].h);
106         }
107         printf("%.2lf\n",ans);
108     }
109     return 0;
110 }

原文地址:https://www.cnblogs.com/letlifestop/p/10335397.html

时间: 2024-08-01 02:35:43

面积并+扫描线 覆盖的面积 HDU - 1255的相关文章

HDU 1255 离散化+扫描线覆盖的面积

覆盖的面积 Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submission(s): 3571    Accepted Submission(s): 1753 Problem Description 给定平面上若干矩形,求出被这些矩形覆盖过至少两次的区域的面积. Input 输入数据的第一行是一个正整数T(1<=T<=100),代表测试数据的数量.每个测试数据

扫描线三巨头 hdu1928&amp;&amp;hdu 1255 &amp;&amp; hdu 1542 [POJ 1151]

学习链接:http://blog.csdn.net/lwt36/article/details/48908031 学习扫描线主要学习的是一种扫描的思想,后期可以求解很多问题. 扫描线求矩形周长并 hdu 1928 Picture Time Limit: 6000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 4795    Accepted Submission(s):

hdu 1255(线段树 扫描线) 覆盖的面积

http://acm.hdu.edu.cn/showproblem.php?pid=1255 典型线段树辅助扫描线,顾名思义扫描线就是相当于yy出一条直线从左到右(也可以从上到下)扫描过去,此时先将所有的横坐标和纵坐标排序 因为是从左到右扫描,那么横坐标应该离散化一下 当扫描线依次扫描的时候,依次扫描到的纵区间在线段树中查找,依据是上边还是下边记录,上边就是-1,下边就是+1, 如果某区间记录值为0的时候,代表没有被覆盖,为1的时候代表覆盖一次,为2代表覆盖两次(不会出现为负数的情况) 最后将依

hdu 1255 覆盖的面积(扫描线)

http://acm.hdu.edu.cn/showproblem.php?pid=1255 一道挺简单的题,让我折腾了许久.主要卡在了更新节点后维护父亲节点上.后来思路明确了就很容易了. 节点信息: l,r:区间端点 cnt:区间被覆盖的次数,cnt = 0说明没有被完全覆盖. len1:区间被覆盖的长度 len2:区间至少被两条线段覆盖的长度. 只要找到父亲节点与子节点在len1,len2,cnt的关系就简单了. #include <stdio.h> #include <iostre

HDU 1255 覆盖的面积 (扫描线 线段树 离散化)

题目链接 题意:中文题意. 分析:纯手敲,与上一道题目很相似,但是刚开始我以为只是把cnt>=0改成cnt>=2就行了,. 但是后来发现当当前加入的线段的范围之前 还有线段的时候就不行了,因为虽然现在都不等于 2,但是之前的那个线段加上现在的已经覆盖2次了. 1 #include <iostream> 2 #include <cstdio> 3 #include <vector> 4 #include <cstring> 5 #include &

hdu 1255 覆盖的面积(线段树&amp;扫描线&amp;重复面积)

覆盖的面积 Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submission(s): 3375    Accepted Submission(s): 1645 Problem Description 给定平面上若干矩形,求出被这些矩形覆盖过至少两次的区域的面积. Input 输入数据的第一行是一个正整数T(1<=T<=100),代表测试数据的数量.每个测试数

hdu 1255 覆盖的面积 线段树扫描线求重叠面积

稀里糊涂打完了没想到1A了,心情还是很舒畅的,c和c++的四舍五入还是四舍六入遇积进位遇偶舍,我感觉很混乱啊,这道题我输出的答案是7.62,也就是遇偶舍了,可是我就手动处理一下进位问题,发现0.005 系统自动进位0.01了,尼玛啊,我一下子就混乱了,不是遇偶舍么,0也是偶数啊,怎么就进位了呢.最后我就不手动处理进位问题了,直接0.2lf%,虽然我输出的结果是7.62,但是提交也过了 这道题和poj1151,hdu1542差不多,扫描线详细讲解http://blog.csdn.net/young

HDU 1255 覆盖的面积 (线段树 + 离散化 + 扫描线)

覆盖的面积 Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submission(s): 4304    Accepted Submission(s): 2139 Problem Description 给定平面上若干矩形,求出被这些矩形覆盖过至少两次的区域的面积. Input 输入数据的第一行是一个正整数T(1<=T<=100),代表测试数据的数量.每个测试数

HDU 1255 覆盖的面积(线段树扫描线)

Problem Description 给定平面上若干矩形,求出被这些矩形覆盖过至少两次的区域的面积. Input 输入数据的第一行是一个正整数T(1<=T<=100),代表测试数据的数量.每个测试数据的第一行是一个正整数N(1<=N<=1000),代表矩形的数量,然后是N行数据,每一行包含四个浮点数,代表平面上的一个矩形的左上角坐标和右下角坐标,矩形的上下边和X轴平行,左右边和Y轴平行.坐标的范围从0到100000. 注意:本题的输入数据较多,推荐使用scanf读入数据. Out