HDU 1542 Atlantis (线段树 + 扫描线 + 离散化)
Atlantis
Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)
Total Submission(s): 8998 Accepted Submission(s): 3856
total area for which maps exist. You (unwisely) volunteered to write a program that calculates this quantity.
not necessarily integers. The values (x1; y1) and (x2;y2) are the coordinates of the top-left resp. bottom-right corner of the mapped area.
The input file is terminated by a line containing a single 0. Don’t process it.
(i.e. the area of the union of all rectangles in this test case), printed exact to two digits to the right of the decimal point.
Output a blank line after each test case.
2
10 10 20 20
15 15 25 25.5
0
Test case #1
Total explored area: 180.00所谓的离散化,大家能够简单的理解为,将一组非常大的数据,浓缩为一组非常小的数据,用这组数据来取代原数据的作用,比方给你1000个数,数的范围为(1,1e18)我们这里就能够用离散化,因为仅仅有1000个数。我们能够用一个数组的下标代表提供的每一数。假设须要这个数据了,因为是下标,能够直接通过下标获得,如此就是离散化。讲得非常基础,是个非常不错的博客然后提醒一下这个扫描线要注意的问题就是区间的问题一般的线段树以及我们的区间改动合并。都有一个共同点,就是不会出现区间缺失的现象,什么叫区间缺失。顾名思义,区间缺失就是缺少一些区间没有进行运算,这里的扫描线就会遇到这个问题。普遍的,我们的线段树以及数据区间分布是这种:[1, a][a + 1, b][b + 1, c][c + 1, d][d + 1, e].......可是假设仅仅是简简单单的用这个来解决扫描线的问题会导致错误,为什么因为。他没有涉及到[a,a + 1],在扫描线中会出现[a,a + 1]中的数据,而经常使用的线段树的区间概念是无法解决这种问题的,出现了所谓的区间缺失,如何解决,以下的代码给出了解决方式,这里简单的提一下,就是利用[ , ),这个区间性质,左闭右开。就可以解决区间缺失问题
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <vector>
#include <queue>
using namespace std;
typedef long long LL;
#define lson rt << 1, l, mid
#define rson rt << 1|1, mid + 1, r
const int MAXN = 2000 + 5;
int Col[MAXN << 2], n, cnt, res;
double X[MAXN << 2], Sum[MAXN << 2];
struct seg {
double l,r,h;
int s;
seg() {}
seg(double l,double r,double h,int s):l(l),r(r),h(h),s(s) {}
bool operator < (const seg & object) const {
return h < object.h;
}
} S[MAXN]; void pushup(int rt,int l,int r) {
if (Col[rt]) Sum[rt] = X[r+1] - X[l];//利用[ , ),这个区间性质。左闭右开
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 rt,int l, int r) {
if(L <= l && r <= R) {
Col[rt] += c;
pushup(rt,l,r);
return ;
}
int mid = (l + r) >> 1;
if(L <= mid) update(L, R, c, lson);
if(R > mid) update(L, R, c, rson);
pushup(rt,l,r);
} int binary_find(double x){
int lb = -1,ub = res - 1;
while(ub - lb > 1){
int mid = (lb + ub) >> 1;
if(X[mid] >= x) ub = mid;
else lb = mid;
}
return ub;
} int main() {
int cas = 1;
while(~ scanf("%d", &n), n) {
cnt = res = 0;
for(int i = 0 ; i < n; i ++) {
double a,b,c,d;
scanf("%lf%lf%lf%lf",&a, &b, &c,&d);
S[cnt] = seg(a, c, b, 1);
X[cnt ++] = a;
S[cnt] = seg(a, c, d, -1);
X[cnt ++] = c;
}
sort(X, X + cnt);
sort(S, S + cnt);
res ++;
for(int i = 1; i < cnt; i ++) {
if(X[i] != X[i - 1]) X[res ++] = X[i];
} memset(Sum, 0, sizeof(Sum));
memset(Col, 0, sizeof(Col));
double ans = 0;
for(int i = 0;i < cnt - 1;i ++){
int l = binary_find(S[i].l);
int r = binary_find(S[i].r) - 1;//利用[ , ),这个区间性质,左闭右开
update(l, r, S[i].s, 1, 0, res - 1);
ans += Sum[1] * (S[i + 1].h - S[i].h);
}
printf("Test case #%d\nTotal explored area: %.2lf\n\n",cas++ , ans);
}
return 0;
}
HDU 1542 Atlantis (线段树 + 扫描线 + 离散化)的更多相关文章
- HDU 1542 - Atlantis - [线段树+扫描线]
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1542 Time Limit: 2000/1000 MS (Java/Others) Memory Li ...
- hdu 1542 Atlantis (线段树扫描线)
大意: 求矩形面积并. 枚举$x$坐标, 线段树维护$[y_1,y_2]$内的边是否被覆盖, 线段树维护边时需要将每条边挂在左端点上. #include <iostream> #inclu ...
- hdu 1542 Atlantis(线段树,扫描线)
Atlantis Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total S ...
- HDU 1542 Atlantis(线段树面积并)
描述 There are several ancient Greek texts that contain descriptions of the fabled island Atlantis. S ...
- POJ 1151 / HDU 1542 Atlantis 线段树求矩形面积并
题意:给出矩形两对角点坐标,求矩形面积并. 解法:线段树+离散化. 每加入一个矩形,将两个y值加入yy数组以待离散化,将左边界cover值置为1,右边界置为2,离散后建立的线段树其实是以y值建的树,线 ...
- Atlantis HDU - 1542 (线段树扫描线)
There are several ancient Greek texts that contain descriptions of the fabled island Atlantis. Some ...
- hdu 1542(线段树+扫描线 求矩形相交面积)
Atlantis Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Su ...
- hdu1542 Atlantis (线段树+扫描线+离散化)
Atlantis Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total S ...
- hdu 1542 Atlantis(段树&扫描线&面积和)
Atlantis Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total S ...
随机推荐
- ALTER USER - 改变数据库用户帐号
SYNOPSIS ALTER USER name [ [ WITH ] option [ ... ] ] where option can be: [ ENCRYPTED | UNENCRYPTED ...
- 常见的HTTP相应状态码
200:请求被正常处理204:请求被受理但没有资源可以返回206:客户端只是请求资源的一部分,服务器只对请求的部分资源执行GET方法,相应报文中通过Content-Range指定范围的资源.301:永 ...
- CAD参数绘制角度标注(com接口)
主要用到函数说明: _DMxDrawX::DrawDimAngular 绘制一个角度标注.详细说明如下: 参数 说明 DOUBLE dAngleVertexX 角度标注的顶点的X值 DOUBLE dA ...
- java.util.MissingResourceException: Can't find bundle for base name db, locale zh_CN
在使用Bundle来加载配置文件的时候, 爆出了这个错误: 原因? 没有找到需要加载的配置文件,因为配置文件必须放在src目录下面, 如果放进了com.bj186.crm的包下面,就必须添加包的名称到 ...
- 18第一章 ASP.Net内建对象
第一章 ASP.Net内建对象 第一章 ASP.Net内建对象 ASP.Net为保持用户的数据和信息,内建了许多对象,包括Application.Response.Requ ...
- Bzoj 3307 雨天的尾巴(线段树合并+树上差分)
C. 雨天的尾巴 题目描述 N个点,形成一个树状结构.有M次发放,每次选择两个点x,y对于x到y的路径上(含x,y)每个点发一袋Z类型的物品.完成所有发放后,每个点存放最多的是哪种物品. 输入格式 第 ...
- linux内核中GNU C和标准C的区别
linux内核中GNU C和标准C的区别 今天看了一下午的linux内核编程方面的内容,发现linux 内核中GNU C与标准C有一些差别,特记录如下: linux 系统上可用的C编译器是GNU C编 ...
- JavaScript在HTML中的应用
JavaScript在HTML中的应用 制作人:全心全意 在HTML文档中可以使用<script>...</script>标记将JavaScript脚本嵌入到其中,在HTML文 ...
- Android使用JDBC连接数据库
连接数据库是安卓开发中几乎不可避免的一项工作,稍有规模的应用通常都需要使用数据库来存储用户数据.对于本地数据库当然可以使用sqlite,而对于多用户线上应用,则一般需要配备云端数据库.其中比较常用且开 ...
- clip-path实现loading圆饼旋转效果以及其他方法
一.loading效果 二.clip-path css中的剪切clip-path属性是CSS Masking模块的一部分. 矩形 clip-path:inset(top right bottom le ...