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

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1255

题意很清楚,就是让你求矩阵之间叠加层数大于1的矩形块的面积和。

因为n只有1000,所以我离散化一下,数据大小就缩小了,那么之后只需要线段树单点更新就好了。

 1 #include <iostream>
 2 #include <cstdio>
 3 #include <cstring>
 4 #include <map>
 5 #include <algorithm>
 6 using namespace std;
 7 const int MAXN = 4e3 + 5;
 8 struct data {
 9     int x1 , x2 , flag;
10     double y , xx1 , xx2;
11     bool operator <(const data& cmp) const {
12         return y < cmp.y;
13     }
14 }a[MAXN];
15 struct segtree {
16     int l , r , add;
17     double val;
18 }T[MAXN << 2];
19 double x[MAXN];
20 map <double , int> mp;
21
22 void build(int p , int l , int r) {
23     int mid = (l + r) >> 1;
24     T[p].val = T[p].add = 0 , T[p].l = l , T[p].r = r;
25     if(r - l == 1) {
26         return ;
27     }
28     build(p << 1 , l , mid);
29     build((p << 1)|1 , mid , r);
30 }
31
32 void updata(int p , int l , int r , int val) {
33     int mid = (T[p].l + T[p].r) >> 1;
34     if(T[p].r - T[p].l == 1) {
35         T[p].add += val;
36         if(T[p].add > 1)
37             T[p].val = x[T[p].r] - x[T[p].l];
38         else
39             T[p].val = 0;
40         return ;
41     }
42     if(r <= mid) {
43         updata(p << 1 , l , r , val);
44     }
45     else if(l >= mid) {
46         updata((p << 1)|1 , l , r , val);
47     }
48     else {
49         updata(p << 1 , l , mid , val);
50         updata((p << 1)|1 , mid , r , val);
51     }
52     T[p].val = T[p << 1].val + T[(p << 1)|1].val;
53 }
54
55 int main()
56 {
57     int t = 1 , n;
58     double x1 , x2 , y1 , y2;
59     scanf("%d" , &t);
60     while(t--) {
61         mp.clear();
62         scanf("%d" , &n);
63         int cnt = 0;
64         for(int i = 0 ; i < n ; i++) {
65             scanf("%lf %lf %lf %lf" , &x1 , &y1 , &x2 , &y2);
66             if(x1 > x2)
67                 swap(x1 , x2);
68             if(y1 > y2)
69                 swap(y1 , y2);
70             int ls = i * 2 , rs = i * 2 + 1;
71             a[ls].xx1 = x1 , a[ls].xx2 = x2 , a[ls].y = y1 , a[ls].flag = 1;
72             a[rs].xx1 = x1 , a[rs].xx2 = x2 , a[rs].y = y2 , a[rs].flag = -1;
73             if(!mp[x1]) {
74                 mp[x1] = 1;
75                 x[++cnt] = x1;
76             }
77             if(!mp[x2]) {
78                 mp[x2] = 1;
79                 x[++cnt] = x2;
80             }
81         }
82         sort(a , a + n * 2);
83         sort(x + 1 , x + cnt + 1);
84         for(int i = 0 ; i < n * 2 ; i++) {
85             a[i].x1 = lower_bound(x + 1 , x + cnt + 1 , a[i].xx1) - x;
86             a[i].x2 = lower_bound(x + 1 , x + cnt + 1 , a[i].xx2) - x;
87         }
88         double res = 0;
89         build(1 , 1 , cnt);
90         updata(1 , a[0].x1 , a[0].x2 , a[0].flag);
91         for(int i = 1 ; i < 2 * n ; i++) {
92             res += (a[i].y - a[i - 1].y) * T[1].val;
93             updata(1 , a[i].x1 , a[i].x2 , a[i].flag);
94         }
95         printf("%.2f\n" , res);
96     }
97 }
时间: 2024-12-24 19:46:43

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

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 覆盖的面积 线段树+扫描线

同 POJ1151 这次是两次 #include <iostream> #include <algorithm> #include <cstdio> #include <cstdlib> #include <cstring> #include <string> #include <stack> #include <queue> #include <cmath> #include <vector

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 覆盖的面积(线段树面积并)

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

hdu1255 覆盖的面积 线段树-扫描线

矩形面积并 线段树-扫描线裸题 1 #include<stdio.h> 2 #include<string.h> 3 #include<algorithm> 4 #include<math.h> 5 using namespace std; 6 const int maxm=2e3+5; 7 const double eps=1e-5; 8 9 struct seg{ 10 double x,y1,y2; 11 int c; 12 bool operator

杭电 HDU ACM 1225 Atlantis (线段树 扫描线 离散化 最基本)

acm第一发扫描线问题,其实算法原理很好理解 ,但是具体实现起来还是遇到诸多问题,我尝试参考了网上两份对于解决 线段树表示区间问题的方法, 第一种是每个结点的真实值,比如对于更新离散化后的1 ~ 4区间,我们在线段树中更新的是1 ~ 3,这样单个结点也可以表示一个最小单位区间. 第二种那就是在建树的时候改变通常策略,1 ~ 10 为总区间,两个孩子为1 ~ 5 ,5 ~ 10. 核心难点:当我们每次找到需要更新的区间,首先应该更新cover值,然后判断此时是被覆盖了,还是解除覆盖了,如果刚好被覆

hdu 5091 Beam Cannon(线段树+扫描线+离散化)

Beam Cannon Time Limit: 3000/1500 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) Total Submission(s): 457    Accepted Submission(s): 175 Problem Description Recently, the γ galaxies broke out Star Wars. Each planet is warring for resou

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

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