POJ 1177/HDU 1828 picture 线段树+离散化+扫描线 轮廓周长计算
求n个图矩形放下来,有的重合有些重合一部分有些没重合,求最后总的不规则图型的轮廓长度。
我的做法是对x进行一遍扫描线,再对y做一遍同样的扫描线,相加即可。因为最后的轮廓必定是由不重合的线段长度组成的,这样理论上是对的
要注意处理高度相同的线段,把底边优先处理(在代码里就是f标记为1的线段),因为若是一个矩形的底边和另一个矩形的上边重合,则这个轮廓肯定不能算
不过POJ和HDU的数据好像都比较弱,我没进行上面的细节处理也AC了,不过一个很简单的数据就会不对,所以还是要处理一下才是真正正确的代码
我之前敲代码的时候还想起在线段树里面的覆盖要涉及懒惰标记(因为是对区间进行覆盖嘛,没向子走)和向上更新。。。不过想了一下不用,这个线段树不是普通那种,因为总是先覆盖底边,然后由等长的上边来解除覆盖,所以区间总是相对不变的,没必要进行向上或者向下更新
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define lson rt<<1,l,mid
#define rson rt<<1|1,mid,r
using namespace std;
const int N=10010;
struct node
{
int lx,ly,hx,hy;
}rec[N];
struct node2
{
int l,r,h,f;
bool operator <(const node2& rhs) const{
if (h==rhs.h){
return f>rhs.f;
}
else return h<rhs.h;
}
}seg[N*2];
int X[N*2];
int n;
int flag[N<<2];
int d[N<<2];
void build(int rt,int l,int r)
{
flag[rt]=0;
d[rt]=0;
if (r-l<=1) return;
int mid=(l+r)>>1;
build(lson);
build(rson);
}
void up(int rt,int l,int r)
{
if (flag[rt]>0){
d[rt]=X[r]-X[l];
}
else
{
if (r-l<=1) d[rt]=0;
else d[rt]=d[rt<<1]+d[rt<<1|1];
}
}
void cover(int L,int R,int v,int rt,int l,int r)
{
if (L<=l && r<=R)
{
flag[rt]+=v;
up(rt,l,r);
return;
}
if (r-l<=1) return ;
int mid=(l+r)>>1;
if (R<=mid) cover(L,R,v,lson);
else
if (L>mid) cover(L,R,v,rson);
else
{
cover(L,R,v,lson);
cover(L,R,v,rson);
}
up(rt,l,r);
}
int main()
{
while (scanf("%d",&n)!=EOF)
{
int cnt=0;
for (int i=0;i<n;i++){
scanf("%d%d%d%d",&rec[i].lx,&rec[i].ly,&rec[i].hx,&rec[i].hy);
X[++cnt]=rec[i].lx;
seg[cnt].l=rec[i].lx;
seg[cnt].r=rec[i].hx;
seg[cnt].h=rec[i].ly;
seg[cnt].f=1; X[++cnt]=rec[i].hx;
seg[cnt].l=rec[i].lx;
seg[cnt].r=rec[i].hx;
seg[cnt].h=rec[i].hy;
seg[cnt].f=-1;
}
int m=1;
sort(X+1,X+1+cnt);
sort(seg+1,seg+1+cnt);
for(int i=2;i<=cnt;i++){
if (X[i]!=X[i-1]){
X[++m]=X[i];
}
}
int ans=0;
build(1,1,m);
//for (int i=1;i<=m;i++) cout<<X[i]<<endl;
for (int i=1;i<=cnt;i++){
//cout<<seg[i].l<<" "<<seg[i].r<<" "<<seg[i].f<<endl;
int l=lower_bound(X+1,X+1+m,seg[i].l)-X;
int r=lower_bound(X+1,X+1+m,seg[i].r)-X;
int tmp=d[1];
//cout<<X[l]<<" xxx "<<X[r]<<endl;
// cout<<"bf: "<<d[1]<<endl;
cover(l,r,seg[i].f,1,1,m);
tmp=d[1]-tmp;
if (tmp<0) tmp=-tmp;
//cout<<"af: "<<d[1]<<endl;
//cout<<tmp<<endl;
ans+=tmp;
}
//cout<<ans<<endl;
cnt=0;
for (int i=0;i<n;i++){
X[++cnt]=rec[i].ly;
seg[cnt].l=rec[i].ly;
seg[cnt].r=rec[i].hy;
seg[cnt].h=rec[i].lx;
seg[cnt].f=1; X[++cnt]=rec[i].hy;
seg[cnt].l=rec[i].ly;
seg[cnt].r=rec[i].hy;
seg[cnt].h=rec[i].hx;
seg[cnt].f=-1;
}
sort(X+1,X+1+cnt);
sort(seg+1,seg+1+cnt);
m=1;
for (int i=2;i<=cnt;i++){
if (X[i]!=X[i-1]){
X[++m]=X[i];
}
}
build(1,1,m);
for (int i=1;i<=cnt;i++){
int l=lower_bound(X+1,X+1+m,seg[i].l)-X;
int r=lower_bound(X+1,X+1+m,seg[i].r)-X;
int tmp=d[1];
cover(l,r,seg[i].f,1,1,m);
tmp=d[1]-tmp;
if (tmp<0) tmp=-tmp;
ans+=tmp;
}
printf("%d\n",ans);
}
return 0;
}
/*
2
0 0 5 2
2 -2 4 3
*/
POJ 1177/HDU 1828 picture 线段树+离散化+扫描线 轮廓周长计算的更多相关文章
- hdu 1828 Picture(线段树 || 普通hash标记)
http://acm.hdu.edu.cn/showproblem.php?pid=1828 Picture Time Limit: 6000/2000 MS (Java/Others) Mem ...
- POJ 1151Atlantis 矩形面积并[线段树 离散化 扫描线]
Atlantis Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 21734 Accepted: 8179 Descrip ...
- HDU 1828 Picture (线段树+扫描线)(周长并)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1828 给你n个矩形,让你求出总的周长. 类似面积并,面积并是扫描一次,周长并是扫描了两次,x轴一次,y ...
- hdu 1828 Picture(线段树轮廓线)
Picture Time Limit: 6000/2000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Sub ...
- HDU 1828 Picture (线段树:扫描线周长)
依然是扫描线,只不过是求所有矩形覆盖之后形成的图形的周长. 容易发现,扫描线中的某一条横边对答案的贡献. 其实就是 加上/去掉这条边之前的答案 和 加上/去掉这条边之后的答案 之差的绝对值 然后横着竖 ...
- POJ1177 Picture 线段树+离散化+扫描线
求最终的覆盖图形周长,写这种代码应该短而精确,差的比较远 /* Problem: 1177 User: 96655 Memory: 348K Time: 32MS Language: C++ Resu ...
- 【POJ 2482】 Stars in Your Window(线段树+离散化+扫描线)
[POJ 2482] Stars in Your Window(线段树+离散化+扫描线) Time Limit: 1000MS Memory Limit: 65536K Total Submiss ...
- hdu1542 矩形面积并(线段树+离散化+扫描线)
题意: 给你n个矩形,输入每个矩形的左上角坐标和右下角坐标. 然后求矩形的总面积.(矩形可能相交). 题解: 前言: 先说说做这道题的感受: 刚看到这道题顿时就懵逼了,几何 烂的渣渣.后来从网上搜题解 ...
- POJ 2528 Mayor's posters(线段树+离散化)
Mayor's posters 转载自:http://blog.csdn.net/winddreams/article/details/38443761 [题目链接]Mayor's posters [ ...
随机推荐
- sigprocmask
sigprocmask 检测和更改进程的信号屏蔽字 初始化信号屏蔽字的函数 sigprocempty--设置空的信号屏蔽字 sigprocfillset----设置全集的信号屏蔽字
- mysql判断数据库或表是否存在
(1) 判断数据库存在, 则删除: drop database if exists db_name;(2) 判断数据表存在, 则删除: drop table if exists table ...
- Decimal为SQL Server、MySql等数据库的一种数据类型
Decimal为SQL Server.MySql等数据库的一种数据类型,不属于浮点数类型,可以在定义时划定整数部份以及小数部分的位数.使用精确小数类型不仅能够保证数据计算更为精确,还可以节省储存空间, ...
- Unity 公告板 Billboard
创建脚本如下 Billboard.cs using UnityEngine; using System.Collections; public class Billboard : MonoBehavi ...
- Django创建完全独立的APP
我们之前已经完成了项目的结构搭建,但是,在Django当中,我们强调的一个重要概念是app,比如Django自带的admin就是一个成功的app典范,那么我们应该如果整理自己的项目结构,才能让我们的a ...
- Window Server 2019 配置篇(8)- 利用MDT定制自动加入域的脚本
我们要更改我们的镜像文件,使得自动安装完成后,计算机已经自动在域中 删除在WDS-server上做的自动部署,我们在上面安装一下MDT和ADK文件,这两个文件如果需要的话可以评论,我会在随后上传 之后 ...
- POJ - 1847 Tram(dijkstra)
题意:有向图有N个点,当电车进入交叉口(某点)时,它只能在开关指向的方向离开. 如果驾驶员想要采取其他方式,他/她必须手动更换开关.当驾驶员从路口A驶向路口B时,他/她尝试选择将他/她不得不手动更换开 ...
- Prometheus Operator 架构【转】
本节讨论 Prometheus Operator 的架构.因为 Prometheus Operator 是基于 Prometheus 的,我们需要先了解一下 Prometheus. Prometheu ...
- oozie的常见错误
1.变量或路径的英文字母写错,常常是大小写搞混,或者是字母顺序颠倒. 2.本地 oozie_works 工作目录下的文件,如job.properties,workflow.xml等,修改后,忘记上传到 ...
- PAN3501与AS3933完美兼容替代
现在不少校园门禁卡都是采用奥地利的AS3933,市场需求是供不应求,当然价格上还是不断上升趋势.成本上压力也是越来越大,不少厂家在寻找能替代软硬件兼容AS3933的芯片方案.今天我就为大家介绍一款能否 ...