Picture

题目链接

http://poj.org/problem?id=1177

Description

A number of rectangular posters, photographs and other pictures of the same shape are pasted on a wall. Their sides are all vertical or horizontal. Each rectangle can be partially or totally covered by the others. The length of the boundary of the union of all rectangles is called the perimeter.

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.

Input

Your program is to read from standard input. The first line contains the number of rectangles pasted on the wall. In each of the subsequent lines, one can find the integer coordinates of the lower left vertex and the upper right vertex of each rectangle. The values of those coordinates are given as ordered pairs consisting of an x-coordinate followed by a y-coordinate.

0 <= number of rectangles < 5000

All coordinates are in the range [-10000,10000] and any existing rectangle has a positive area.

Output

Your program is to write to standard output. The output must contain a single line with a non-negative integer which corresponds to the perimeter for the input rectangles.

Sample Input

    7
-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

Sample Output

    228

题意

经典入门题:求矩形周长并。

题解

我不知道别人怎么做的,我要扫四边,上下左右各扫一遍,具体说说从下往上扫,没扫到一条下边,就判断他下面有没有下边遮挡,减去遮挡长度就是这条下边所能贡献的有效长度;遇到上边,就把对应的下边去掉就好了。如果学过矩形面积并的话,应该很好理解。

代码

#include<cstdio>
#include<algorithm>
#include<string>
using namespace std;
#define ll long long
#define INF 0x7f7f7f7f
#define N 100050
ll ans;
int n,m,kth[N],cnt1,cnt2,num;
template<typename T>void read(T&x)
{
ll k=0; char c=getchar();
x=0;
while(!isdigit(c)&&c!=EOF)k^=c=='-',c=getchar();
if (c==EOF)exit(0);
while(isdigit(c))x=x*10+c-'0',c=getchar();
x=k?-x:x;
}
void read_char(char &c)
{while(!isalpha(c=getchar())&&c!=EOF);}
struct Query
{
int l,r,h,id;
bool operator<(const Query&e)const
{return h<e.h;};
}que[3][N];
struct Node{int l,r,lazy,sum;};
struct segmentTree
{
Node tr[N<<2];
void push_up(int x);
void bt(int x,int l,int r);
void update(int x,int l,int r,int tt);
int query(int x,int l,int r);
}seg;
void segmentTree::push_up(int x)
{
int len=kth[tr[x].r+1]-kth[tr[x].l];
if (tr[x].l<tr[x].r)tr[x].sum=tr[x<<1].sum+tr[x<<1|1].sum;
else tr[x].sum=0;
if (tr[x].lazy>0)tr[x].sum=len;
}
void segmentTree::bt(int x,int l,int r)
{
tr[x]={l,r,0,0};
if (l==r)return;
int mid=(l+r)>>1;
bt(x<<1,l,mid);
bt(x<<1|1,mid+1,r);
}
void segmentTree::update(int x,int l,int r,int tt)
{
if (l<=tr[x].l&&tr[x].r<=r)
{
tr[x].lazy+=tt;
push_up(x);
return;
}
int mid=(tr[x].l+tr[x].r)>>1;
if(l<=mid)update(x<<1,l,r,tt);
if(mid<r)update(x<<1|1,l,r,tt);
push_up(x);
}
int segmentTree::query(int x,int l,int r)
{
if (l<=tr[x].l&&tr[x].r<=r)return tr[x].sum;
if (tr[x].lazy>0)return kth[r+1]-kth[l];
int mid=(tr[x].l+tr[x].r)>>1,ans=0;
if (l<=mid)ans+=query(x<<1,max(tr[x].l,l),min(mid,r));
if (mid<r)ans+=query(x<<1|1,max(mid+1,l),min(tr[x].r,r));
return ans;
}
void solve(Query *que,int cnt)
{
seg.bt(1,1,num);
sort(que+1,que+cnt+1);
for(int i=1;i<=cnt-1;i++)
{
int l=lower_bound(kth+1,kth+num+1,que[i].l)-kth;
int r=lower_bound(kth+1,kth+num+1,que[i].r)-kth-1;
if (que[i].id==1)
ans+=1LL*(kth[r+1]-kth[l]-seg.query(1,l,r));
seg.update(1,l,r,que[i].id);
}
}
void work()
{
n=20050;
read(m);
for(int i=1;i<=m;i++)
{
int x1,y1,x2,y2;
read(x1); read(y1); read(x2); read(y2);
que[1][++cnt1]=Query{x1,x2,y1,1};
que[1][++cnt1]=Query{x1,x2,y2,-1};
que[2][++cnt2]=Query{y1,y2,x1,1};
que[2][++cnt2]=Query{y1,y2,x2,-1};
kth[++num]=x1;
kth[++num]=x2;
kth[++num]=y1;
kth[++num]=y2;
}
sort(kth+1,kth+num+1);
num=unique(kth+1,kth+num+1)-kth-1;//
solve(que[1],cnt1);
solve(que[2],cnt2);
for(int i=1;i<=cnt1;i++)que[1][i].h=-que[1][i].h,que[1][i].id=-que[1][i].id;
for(int i=1;i<=cnt2;i++)que[2][i].h=-que[2][i].h,que[2][i].id=-que[2][i].id;
solve(que[1],cnt1);
solve(que[2],cnt2); printf("%lld\n",ans);
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("aa.in","r",stdin);
#endif
work();
}

POJ 1177 矩形周长并 模板的更多相关文章

  1. HDU 1828 / POJ 1177 Picture --线段树求矩形周长并

    题意:给n个矩形,求矩形周长并 解法:跟求矩形面积并差不多,不过线段树节点记录的为: len: 此区间线段长度 cover: 此区间是否被整个覆盖 lmark,rmark: 此区间左右端点是否被覆盖 ...

  2. HDU 1828 / POJ 1177 Picture (线段树扫描线,求矩阵并的周长,经典题)

    做这道题之前,建议先做POJ 1151  Atlantis,经典的扫描线求矩阵的面积并 参考连接: http://www.cnblogs.com/scau20110726/archive/2013/0 ...

  3. HDU 1828 POJ 1177 Picture

    矩形周长并 POJ上C++,G++都能过,HDU上C++过了,G++WA ,不知道为什么 #include<cstdio> #include<cstring> #include ...

  4. POJ - 1177 线段树

    POJ - 1177 扫描线 这道题也算是一道扫描线的经典题目了. 只不过这道题是算周长,非常有意思的一道题.我们已经知道了,一般求面积并,是如何求的,现在我们要把扫描线进行改造一下,使得能算周长. ...

  5. 25.按要求编写一个Java应用程序: (1)编写一个矩形类Rect,包含: 两个属性:矩形的宽width;矩形的高height。 两个构造方法: 1.一个带有两个参数的构造方法,用于将width和height属性初化; 2.一个不带参数的构造方法,将矩形初始化为宽和高都为10。 两个方法: 求矩形面积的方法area() 求矩形周长的方法perimeter() (2)通过继承Rect类编写一个具有

    package zhongqiuzuoye; //自己写的方法 public class Rect { public double width; public double height; Rect( ...

  6. HDU 1828 扫描线(矩形周长并)

    Picture Time Limit: 6000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Sub ...

  7. poj 1113 凸包周长

    Wall Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 33888   Accepted: 11544 Descriptio ...

  8. HDU 6362(求椭圆中矩形周长的期望 数学)

    题意是给定一个椭圆标准方程的a,b(椭圆的长半轴长和短半轴长),在[0,b]内取一个数,则过点(0,b)且平行于x轴的直线与椭圆交于两点,再将此两点关于x轴做对称点,顺次连接此四点构成矩形,求出这些矩 ...

  9. hdu 1828 Picture(线段树扫描线矩形周长并)

    线段树扫描线矩形周长并 #include <iostream> #include <cstdio> #include <algorithm> #include &l ...

随机推荐

  1. 【概率论】1-1:概率定义(Definition of Probability)

    title: [概率论]1-1:概率定义(Definition of Probability) categories: Mathematic Probability keywords: Sample ...

  2. 实战 Prometheus 搭建监控系统

    实战 Prometheus 搭建监控系统 Prometheus 是一款基于时序数据库的开源监控告警系统,说起 Prometheus 则不得不提 SoundCloud,这是一个在线音乐分享的平台,类似于 ...

  3. java 构造方法中super()和this()

    1)super(参数):调用基类中的某一个构造函数(应该为构造函数中的第一条语句) 2)this(参数):调用本类中另一种形成的构造函数(应该为构造函数中的第一条语句) 3)super: 它引用当前对 ...

  4. CF1214B

    CF1214B 解法: 暴力枚举,时间复杂度 $ O(n) $ CODE: #include<iostream> #include<cstdio> #include<cs ...

  5. mysql启动关闭脚本

    #!/bin/sh mysql_port= mysql_username="root" mysql_password="" function_start_mys ...

  6. Linux上命令行检出、提交和更新操作

    1.创建工作区目录 列:我创建两个工作目录,用来模拟两个开发人员,命令如下:(工作路径可以按照自己需要随意改变) mkdir -p /root/workspace/harry mkdir -p /ro ...

  7. Linux设备驱动程序 之 延迟执行

    长延迟 有些驱动程序需要延迟比较长的时间,即长于一个时钟滴答: 忙等待 如果想把执行延迟若干个时钟滴答,或者对延迟的精度要求不高,最简单的实现方法就是一个监视jiffies计数器的循环:这种忙等待的实 ...

  8. 组件 computed 与 vuex 中 getters 的使用,及 mapGetters 的使用,对象上追加属性,合并对象

    vue 是响应式的数据,这一点相当的方便我们的操作,但有些错误的操作方法会 vue 的响应无效 除此之外我们还要了解 vue.set() 和 Object.assgin() 的使用 vue.set() ...

  9. Flutter移动电商实战 --(16)切换后页面状态的保持AutomaticKeepAliveClientMixin

    底栏切换每次都重新请求是一件非常恶心的事,flutter 中提供了AutomaticKeepAliveClientMixin 帮我们完成页面状态保存效果. 1.AutomaticKeepAliveCl ...

  10. office很抱歉遇到一些临时服务器问题

    office2016登录很抱歉遇到一些临时服务器问题   主要问题:word不能进行发博客了.一直以为是cnblog服务器不稳定,今天才发现,word不能登录也就是不能联网. 查了原因,才知道是代理造 ...