求矩形面积并,离散化加线段树。

扫描线法:

用平行x轴的直线扫,每次ans+=(下一个高度-当前高度)*当前覆盖的宽度。

#include<algorithm>
#include<cstdio>
#include<cstring>
#define dd double
#define ll long long
#define N 201
using namespace std;
struct P{dd s,e,h;int f;}p[N];
struct Tree{dd sum;int c;}t[N<<];
dd a,b,c,d,x[N],ans;
int n,m,num;
int cmp(const P &a,const P &b){
return a.h<b.h;
}
void pushUp(ll rt,ll l,ll r){
if(t[rt].c)t[rt].sum=x[r+]-x[l];//r+1是因为节点[l,r]表示区间[x[l],x[r+1]]。
else if(l==r)t[rt].sum=;
else t[rt].sum=t[rt<<].sum+t[rt<<|].sum;
}
void update(ll s,ll e,ll rt,ll l,ll r,ll v){
if(s<=l&&r<=e) t[rt].c+=v;
else {
if(l>e||r<s)return;
ll m=l+r>>;
update(s,e,rt<<,l,m,v);
update(s,e,rt<<|,m+,r,v);
}
pushUp(rt,l,r);
}
int main()
{
while(scanf("%d",&n),n){
int k=;
for(int i=;i<n;i++){
scanf("%lf%lf%lf%lf",&a,&b,&c,&d);
x[++k]=a,p[k]=(P){a,c,b,};
x[++k]=c,p[k]=(P){a,c,d,-};
}
sort(x+,x++k);
sort(p+,p++k,cmp);
m=;
for(int i=;i<=k;i++)
if(x[i]!=x[i-])x[m++]=x[i];
ans=;
for(int i=;i<=k;i++){//共k条线段,每次计算p[i].h到p[i+1].h之间的面积,第k次相当于清空所有,酱就不用初始化线段树了。
int l=lower_bound(x,x+m,p[i].s)-x;
int r=lower_bound(x,x+m,p[i].e)-x-;//r-1同上原因
update(l,r,,,m-,p[i].f);
ans+=t[].sum*(p[i+].h-p[i].h);
}
printf("Test case #%d\nTotal explored area: %.2lf\n\n",++num,ans);
}
return ;
}

另一种方法还是线段树,这里扫描线用的是平行y轴的直线,每次增加的面积是当前扫描的竖线所在的高度区间的最后一次的x与当前x的差值乘上区间的高度。所以每次增加的不一定是一个矩形,而是多个矩形并。

 

#include<cstdio>
#include<algorithm>
#define dd double
using namespace std;
#define N 201
struct P{dd x,y1,y2;int f;}p[N];
struct TREE{dd y1,y2,x;int c,f;}tree[N<<];
dd x1,y1,x2,y2,y[N];
int n,k,num;
int cmp(const P &a,const P &b){
return a.x<b.x;
}
void build(int i,int l,int r){
tree[i].c=tree[i].f=;
tree[i].y1=y[l];//直接将线段树节点代表的区间存在线段树里
tree[i].y2=y[r];
if(l+==r){
tree[i].f=;
return;
}
int mk=(l+r)>>;
build(i<<,l,mk);
build(i<<|,mk,r);
}
dd insert(int i,dd x,dd l,dd r,int flag){
if(r<=tree[i].y1||l>=tree[i].y2)
return ;
if(tree[i].f){//离散后的一个最小区间,叶子节点
dd ans;
if(tree[i].c>)//全覆盖
ans=(x-tree[i].x)*(tree[i].y2-tree[i].y1);//(当前x-该区间最后的x)*区间高度
else
ans=;
tree[i].x=x;//该区间最新的x
tree[i].c+=flag;//更新覆盖
return ans;
}
return insert(i<<,x,l,r,flag)+insert(i<<|,x,l,r,flag);
}
int main(){
while(scanf("%d",&n),n){
k=;
for(int i=;i<=n;i++){
scanf("%lf%lf%lf%lf",&x1,&y1,&x2,&y2);
y[++k]=y1,p[k]=(P){x1,y1,y2,};
y[++k]=y2,p[k]=(P){x2,y1,y2,-};
}
sort(y+,y+k+);
sort(p+,p+k+,cmp);
//没有去重,其实数量少,没必要。
build(,,k);
dd ans=;
for(int i=;i<=k;i++)
ans+=insert(,p[i].x,p[i].y1,p[i].y2,p[i].f);
printf("Test case #%d\nTotal explored area: %.2f\n\n",++num,ans);
}
return ;
}

  

【HDU 1542】Atlantis(线段树+离散化,矩形面积并)的更多相关文章

  1. POJ 1151 / HDU 1542 Atlantis 线段树求矩形面积并

    题意:给出矩形两对角点坐标,求矩形面积并. 解法:线段树+离散化. 每加入一个矩形,将两个y值加入yy数组以待离散化,将左边界cover值置为1,右边界置为2,离散后建立的线段树其实是以y值建的树,线 ...

  2. POJ 1151 Atlantis 线段树求矩形面积并 方法详解

    第一次做线段树扫描法的题,网搜各种讲解,发现大多数都讲得太过简洁,不是太容易理解.所以自己打算写一个详细的.看完必会o(∩_∩)o 顾名思义,扫描法就是用一根想象中的线扫过所有矩形,在写代码的过程中, ...

  3. HDU 1542.Atlantis-线段树求矩形面积并(离散化、扫描线/线段树)-贴模板

    好久没写过博客了,这学期不是很有热情去写博客,写过的题也懒得写题解.现在来水一水博客,写一下若干年前的题目的题解. Atlantis Time Limit: 2000/1000 MS (Java/Ot ...

  4. HDU 1542 Atlantis(线段树面积并)

     描述 There are several ancient Greek texts that contain descriptions of the fabled island Atlantis. S ...

  5. HDU 1542 Atlantis (线段树 + 扫描线 + 离散化)

    Atlantis Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total S ...

  6. HDU 1542 - Atlantis - [线段树+扫描线]

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1542 Time Limit: 2000/1000 MS (Java/Others) Memory Li ...

  7. HDU 1542 Atlantics 线段树+离散化扫描

    将 x 轴上的点进行离散化,扫描线沿着 y 轴向上扫描 每次添加一条边不断找到当前状态有效边的长度 , 根据这个长度和下一条边形成的高度差得到一块合法的矩形的面积 #include<iostre ...

  8. hdu 1542 Atlantis(线段树,扫描线)

    Atlantis Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total S ...

  9. hdu 1542 Atlantis (线段树扫描线)

    大意: 求矩形面积并. 枚举$x$坐标, 线段树维护$[y_1,y_2]$内的边是否被覆盖, 线段树维护边时需要将每条边挂在左端点上. #include <iostream> #inclu ...

  10. HDU 1255 覆盖的面积 (扫描线 线段树 离散化 矩形面积并)

    题目链接 题意:中文题意. 分析:纯手敲,与上一道题目很相似,但是刚开始我以为只是把cnt>=0改成cnt>=2就行了,. 但是后来发现当当前加入的线段的范围之前 还有线段的时候就不行了, ...

随机推荐

  1. [No00000F]Excel快捷键大全 Excel2013/2010/2007/2003常用快捷键大全

    一个软件最大的用处是提高工作效率,衡量一个软件的好坏,除了是否出名之外,最主就是能否让一个新手更快的学会这个软件和提高工作速度.就拿Excel表格来说吧,平常办公中我们经常会用它来制作表格,统计数据或 ...

  2. java 27 - 6 反射之 通过配置文件运行类中的方法

    在以前,如果我们想要调用一个类中的方法,只能这样子: 例: 有Cat和Dog两个类,里面有eat和run两个成员方法: public class Dog { public void eat() { S ...

  3. Android Fragment 你应该知道的一切

    转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/42628537,本文出自:[张鸿洋的博客] 很久以前写过两篇Fragment的介绍 ...

  4. jsonobject 遍历 org.json.JSONObject

    import org.json.JSONArray; import org.json.JSONException; import org.json.JSONObject; public static  ...

  5. Android之监听手机软键盘弹起与关闭

    背景: 在很多App开发过程中需要在Activity中监听Android设备的软键盘弹起与关闭,但是Android似乎没有提供相关的的监听API给我们来调用,本文提供了一个可行的办法来监听软键盘的弹起 ...

  6. 将pdf文件通过itunes直接拖到ipad的ibooks里面

    开始不太清楚进行过什么设置,使得以前可以直接通过拖动的方式复制pdf文件到ipad里面的方法不管用了.在帖子http://bbs.weiphone.com/read-htm-tid-864091-pa ...

  7. 我心目中的Asp.net核心对象

    转:http://www.cnblogs.com/fish-li/archive/2011/08/21/2148640.html 阅读目录 开始 HttpRuntime HttpServerUtili ...

  8. linux系统下对网站实施负载均衡+高可用集群需要考虑的几点

    随着linux系统的成熟和广泛普及,linux运维技术越来越受到企业的关注和追捧.在一些中小企业,尤其是牵涉到电子商务和电子广告类的网站,通常会要求作负载均衡和高可用的Linux集群方案. 那么如何实 ...

  9. 从浏览器输入url到页面加载完成都发生了什么

    一个http请求的过程 简要介绍一下一个http请求的网络传输过程: DNS Lookup先获得URL对应的IP地址 Socket Connect浏览器和服务器建立TCP连接 Send Request ...

  10. 前端面试——css篇

    css盒子模型 在W3C模型中: 总宽度 = margin-left + border-left + padding-left + width + padding-right + border-rig ...