扫描线三巨头 hdu1928&&hdu 1255 && 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): 2339
Write a program to calculate the perimeter. An example with 7 rectangles is shown in Figure 1.

The corresponding boundary is the whole set of line segments drawn in Figure 2.

The vertices of all rectangles have integer coordinates.
0 <= number of rectangles < 5000
All coordinates are in the range [-10000,10000] and any existing rectangle has a positive area.
Please process to the end of file.
-15 0 5 10
-5 8 20 25
15 -4 24 14
0 -6 16 4
2 15 10 22
30 10 36 20
34 0 40 16
//#pragma comment(linker, "/STACK:1024000000,1024000000")
#include<cstdio>
#include<iostream>
#include<cstring>
#include<algorithm>
#include<cmath>
#define clr(x) memset(x,0,sizeof(x))
#define MAXN 50010
using namespace std;
struct edgx
{
int l,u,x;
int d;
}edgex[MAXN];
struct edgy
{
int l,r,y;
int d;
}edgey[MAXN];
struct seg
{
int l,r,cov,len;
}segt[MAXN<<];
int cntx,cnty;
int x[MAXN],y[MAXN],vec[MAXN];
bool cmpy(edgy a,edgy b)
{
if(a.y==b.y) return a.d>b.d;
return a.y<b.y;
}
bool cmpx(edgx a,edgx b)
{
if(a.x==b.x) return a.d>b.d;
return a.x<b.x;
}
void init(int i,int l,int r)
{
segt[i]=(seg){l,r,,};
if(l==r)
return ;
int mid=(l+r)>>;
init(i<<,l,mid);
init(i<<|,mid+,r);
return ;
}
void pushup(int i)
{
if(segt[i].cov)
{
segt[i].len=vec[segt[i].r+]-vec[segt[i].l];
}
else if(segt[i].l==segt[i].r)
{
segt[i].len=;
}
else
{
segt[i].len=segt[i<<].len+segt[i<<|].len;
}
return ;
}
void update(int i,int l,int r,int value)
{
if(segt[i].l>=l && segt[i].r<=r)
{
segt[i].cov+=value;
pushup(i);
return ;
}
int mid=(segt[i].l+segt[i].r)>>;
if(mid>=r)
{
update(i<<,l,r,value);
}
else if(mid<l)
{
update(i<<|,l,r,value);
}
else
{
update(i<<,l,r,value);
update(i<<|,l,r,value);
}
pushup(i);
return ;
}
int main()
{
int x1,x2,y1,y2,n,m,T,ans,l,r,k;
while(scanf("%d",&n)!=EOF)
{
cntx=;
cnty=;
for(int i=;i<=n;i++)
{
scanf("%d%d%d%d",&x1,&y1,&x2,&y2);
edgex[++cntx]=(edgx){y1,y2,x1,};
x[cntx]=x1;
edgex[++cntx]=(edgx){y1,y2,x2,-};
x[cntx]=x2;
edgey[++cnty]=(edgy){x1,x2,y1,};
y[cnty]=y1;
edgey[++cnty]=(edgy){x1,x2,y2,-};
y[cnty]=y2;
}
n<<=;
ans=;
memcpy(vec,x,sizeof(x));
sort(vec+,vec+n+);
m=unique(vec+,vec+n+)-vec-;
sort(edgey+,edgey+n+,cmpy);
init(,,m);
for(int i=;i<=n;i++)
if(edgey[i].l<edgey[i].r)
{
k=segt[].len;
l=lower_bound(vec+,vec+m+,edgey[i].l)-vec;
r=lower_bound(vec+,vec+m+,edgey[i].r)-vec;
update(,l,r-,edgey[i].d);
ans+=abs(segt[].len-k);
}
memcpy(vec,y,sizeof(y));
sort(vec+,vec+n+);
m=unique(vec+,vec+n+)-vec-;
sort(edgex+,edgex+n+,cmpx);
init(,,m);
for(int i=;i<=n;i++)
if(edgex[i].l<edgex[i].u)
{
k=segt[].len;
l=lower_bound(vec+,vec+m+,edgex[i].l)-vec;
r=lower_bound(vec+,vec+m+,edgex[i].u)-vec;
update(,l,r-,edgex[i].d);
ans+=abs(segt[].len-k);
}
printf("%d\n",ans);
}
return ;
}
hdu 1255 矩阵面积交
覆盖的面积
Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)
Total Submission(s): 5718 Accepted Submission(s): 2854

注意:本题的输入数据较多,推荐使用scanf读入数据.
5
1 1 4 2
1 3 3 7
2 1.5 5 4.5
3.5 1.25 7.5 4
6 3 10 7
3
0 0 1 1
1 0 2 1
2 0 3 1
0.00
//#pragma comment(linker, "/STACK:1024000000,1024000000")
#include<cstdio>
#include<iostream>
#include<cstring>
#include<algorithm>
#include<cmath>
#define clr(x) memset(x,0,sizeof(x))
#define MAXN 10010
using namespace std;
struct edg
{
double l,r,y;
int d;
}edge[MAXN];
struct seg
{
int l,r,cov;
double len1,len2;
}segt[MAXN<<];
int cnt;
double x[MAXN];
bool cmp(edg a,edg b)
{
if(a.y==b.y) return a.d>b.d;
return a.y<b.y;
}
double max(double a,double b)
{
return a>b?a:b;
}
void init(int i,int l,int r)
{
segt[i]=(seg){l,r,,,};
if(l==r)
return ;
int mid=(l+r)>>;
init(i<<,l,mid);
init(i<<|,mid+,r);
return ;
}
void pushup(int i)
{
if(segt[i].cov>=)
{
segt[i].len2=segt[i].len1=x[segt[i].r+]-x[segt[i].l];
}
else if(segt[i].cov==)
{
segt[i].len1=x[segt[i].r+]-x[segt[i].l];
if(segt[i].l==segt[i].r)
segt[i].len2=;
else
segt[i].len2=max(segt[i<<].len1,segt[i<<].len2)+max(segt[i<<|].len1,segt[i<<|].len2);
}
else
{
if(segt[i].l==segt[i].r)
{
segt[i].len1=segt[i].len2=;
}
else
{
segt[i].len2=segt[i<<].len2+segt[i<<|].len2;
segt[i].len1=segt[i<<].len1+segt[i<<|].len1;
}
}
return ;
}
void update(int i,int l,int r,int value)
{
if(segt[i].l>=l && segt[i].r<=r)
{
segt[i].cov+=value;
pushup(i);
return ;
}
int mid=(segt[i].l+segt[i].r)>>;
if(mid>=r)
{
update(i<<,l,r,value);
}
else if(mid<l)
{
update(i<<|,l,r,value);
}
else
{
update(i<<,l,r,value);
update(i<<|,l,r,value);
}
pushup(i);
return ;
}
int main()
{
int T,n,m,k,u,v;
double x1,x2,y1,y2,ans,l,r;
scanf("%d",&T);
while(T--)
{
scanf("%d",&n);
cnt=;
ans=;
for(int i=;i<=n;i++)
{
scanf("%lf%lf%lf%lf",&x1,&y1,&x2,&y2);
edge[++cnt]=(edg){x1,x2,y1,};
x[cnt]=x1;
edge[++cnt]=(edg){x1,x2,y2,-};
x[cnt]=x2;
}
n<<=;
sort(x+,x+n+);
m=unique(x+,x+n+)-x-;
sort(edge+,edge+n+,cmp);
init(,,m);
for(int i=;i<n;i++)
if(edge[i].r>edge[i].l)
{
l=lower_bound(x+,x+m+,edge[i].l)-x;
r=lower_bound(x+,x+m+,edge[i].r)-x;
update(,l,r-,edge[i].d);
ans+=segt[].len2*(edge[i+].y-edge[i].y);
}
printf("%0.2lf\n",ans);
}
return ;
}
hdu 1542 [POJ 1151] 区间面积并
Atlantis
Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)
Total Submission(s): 12537 Accepted Submission(s): 5257
The input file is terminated by a line containing a single 0. Don’t process it.
Output a blank line after each test case.
10 10 20 20
15 15 25 25.5
0
Total explored area: 180.00
//#pragma comment(linker, "/STACK:1024000000,1024000000")
#include<cstdio>
#include<iostream>
#include<cstring>
#include<algorithm>
#include<cmath>
#define clr(x) memset(x,0,sizeof(x))
#define MAXN 10010
using namespace std;
struct edg
{
double l,r,y;
int d;
}edge[MAXN];
struct seg
{
int l,r,cov;
double len;
}segt[MAXN<<];
int cnt;
double x[MAXN];
bool cmp(edg a,edg b)
{
if(a.y==b.y) return a.d>b.d;
return a.y<b.y;
}
double max(double a,double b)
{
return a>b?a:b;
}
void init(int i,int l,int r)
{
segt[i]=(seg){l,r,,};
if(l==r)
return ;
int mid=(l+r)>>;
init(i<<,l,mid);
init(i<<|,mid+,r);
return ;
}
void pushup(int i)
{
if(segt[i].cov)
{
segt[i].len=x[segt[i].r+]-x[segt[i].l];
}
else if(segt[i].l==segt[i].r)
{
segt[i].len=;
}
else
{
segt[i].len=segt[i<<].len+segt[i<<|].len;
}
return ;
}
void update(int i,int l,int r,int value)
{
if(segt[i].l>=l && segt[i].r<=r)
{
segt[i].cov+=value;
pushup(i);
return ;
}
int mid=(segt[i].l+segt[i].r)>>;
if(mid>=r)
{
update(i<<,l,r,value);
}
else if(mid<l)
{
update(i<<|,l,r,value);
}
else
{
update(i<<,l,r,value);
update(i<<|,l,r,value);
}
pushup(i);
return ;
}
int main()
{
int T,n,m,k,u,v;
double x1,x2,y1,y2,ans,l,r;
int kase=;
while(scanf("%d",&n) && n!=)
{
printf("Test case #%d\n",++kase);
cnt=;
ans=;
for(int i=;i<=n;i++)
{
scanf("%lf%lf%lf%lf",&x1,&y1,&x2,&y2);
edge[++cnt]=(edg){x1,x2,y1,};
x[cnt]=x1;
edge[++cnt]=(edg){x1,x2,y2,-};
x[cnt]=x2;
}
n<<=;
sort(x+,x+n+);
m=unique(x+,x+n+)-x-;
sort(edge+,edge+n+,cmp);
init(,,m);
for(int i=;i<n;i++)
if(edge[i].r>edge[i].l)
{
l=lower_bound(x+,x+m+,edge[i].l)-x;
r=lower_bound(x+,x+m+,edge[i].r)-x;
update(,l,r-,edge[i].d);
ans+=segt[].len*(edge[i+].y-edge[i].y);
}
printf("Total explored area: %0.2lf\n",ans);
printf("\n");
}
return ;
}
扫描线三巨头 hdu1928&&hdu 1255 && hdu 1542 [POJ 1151]的更多相关文章
- hdu 1542&&poj 1151 Atlantis[线段树+扫描线求矩形面积的并]
Atlantis Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total S ...
- hdu 1542 & & poj 1151
Atlantis Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total S ...
- HDU 1542/POJ 1151 Atlantis (scaning line + segment tree)
A template of discretization + scaning line + segment tree. It's easy to understand, but a little di ...
- 线段树扫描线(一、Atlantis HDU - 1542(覆盖面积) 二、覆盖的面积 HDU - 1255(重叠两次的面积))
扫描线求周长: hdu1828 Picture(线段树+扫描线+矩形周长) 参考链接:https://blog.csdn.net/konghhhhh/java/article/details/7823 ...
- HDU - 1255 覆盖的面积(线段树求矩形面积交 扫描线+离散化)
链接:线段树求矩形面积并 扫描线+离散化 1.给定平面上若干矩形,求出被这些矩形覆盖过至少两次的区域的面积. 2.看完线段树求矩形面积并 的方法后,再看这题,求的是矩形面积交,类同. 求面积时,用被覆 ...
- HDU 1255 覆盖的面积(线段树+扫描线)
题目地址:HDU 1255 这题跟面积并的方法非常像,仅仅只是须要再加一个变量. 刚開始我以为直接用那个变量即可,仅仅只是推断是否大于0改成推断是否大于1.可是后来发现了个问题,由于这个没有下放,没延 ...
- hdu 1255 覆盖的面积(线段树 面积 交) (待整理)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1255 Description 给定平面上若干矩形,求出被这些矩形覆盖过至少两次的区域的面积. In ...
- hdu 1255 覆盖的面积(求覆盖至少两次以上的面积)
了校赛,还有什么途径可以申请加入ACM校队? 覆盖的面积 Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65536/32768 K ...
- “三巨头”有变化,BAT还能走多久?
在腾讯市值超越阿里巴巴后,市场分析多数认为,当年的BAT“三巨头”时代已经彻底结束,进入了“双寡头”时代了 从对外投资来看,BAT不同的投资逻辑可以推测其战略方向 撰文/梁云风 时评员,关注财经与互联 ...
随机推荐
- .NET中使用switch和java不一样的地方。
1.不能这样贯穿 我们知道,java 和 C在使用switch时候可以这样. switch (i) { //java中此处不使用break // 执行了case 1:对应的语句后直接 贯穿到 case ...
- idea如何搭建springmvc4
1.推荐大牛博客 此操作我操作了三次过后终于成功了,奉献大牛博客连接:做的非常详细到位,望各位采纳,推荐置顶. https://www.cnblogs.com/chenlinghong/p/83395 ...
- bzoj 3453 数论
首先我们知道对于f(x)来说,它是一个k次的多项式,那么f(x)的通项公式可以表示成一个k+1次的式子,且因为f(x)没有常数项,所以我们设这个式子为 f(x)=Σ(a[i]*x^i) (1<= ...
- Android使用TextView实现跑马灯效果(自定义控件)
对于一个长的TetxView 折行显示是一个很好的办法,另一种方法就是跑马灯显示(单行滚动) 1.折行显示的长TextView <LinearLayout xmlns:android=" ...
- Java多线程学习(三)volatile关键字
转载请备注地址:https://blog.csdn.net/qq_34337272/article/details/79680693 系列文章传送门: Java多线程学习(一)Java多线程入门 Ja ...
- Perl6 Bailador框架(1):开始
use v6; use Bailador; get '/' => sub { '<h1><center>Hello, World</center></h ...
- 一个python拖库字段的小脚本
import requests import re all_column = dict() all_db = "db_zf,dg_activity,dg_activity_log,dg_ad ...
- java===java基础学习(11)---继承
继承可以解决代码复用,让编程更加靠近人的思维.当多个类存在相同的属性(变量)和方法时,可以从这些类中抽象出父类,在父类中定义这些相同的属性和方法.所有的子类不需要重新定义这些属性和方法,只需要通过ex ...
- 神秘的subsys_initcall【转】
转自:http://blog.chinaunix.net/uid-12567959-id-161015.html 在内核代码里到处都能看到这个subsys_initcall(),而它到底是干什么的呢? ...
- vscode和phpStorm使用xdebug调试设置
phpStorm http://www.cnblogs.com/cxscode/p/7045944.html http://www.cnblogs.com/cxscode/p/7050781.html ...