poj1177
题意:在平面直角坐标系内给出一些与坐标轴平行的矩形,将这些矩形覆盖的区域求并集,然后问这个区域的周长是多少。(边与边重合的地方不计入周长)
分析:线段树。曾经做过类似的求矩形覆盖的总面积的题。这道题同样要使用扫描线算法。属于线保留型线段树。
我们先领扫描线与y轴平行。
线段树内每个节点除了要记录该区间被覆盖了几层之外,还要记录当前状态下扫描线在该区间(开区间)内与多少条与x轴平行的边相交。
节点上还有两个bool型变量,记录该区间内(包括子树)线段覆盖是否接触到该区间的起始和结束点。
在父节点如果没被整个覆盖,则需要从子区间的起始和结束点来更新父节点两端点的覆盖情况。
更新过程在返回时,父节点的交点数量应等于两子节点交点数量的和,另外特判一下两子区间的公共点(父节点的中点),判断这里是不是覆盖与未覆盖的分界点,如果是则还需要在父节点上增加这个交点。
这样就得知了每段与x轴平行的距离内有多少条线段需要计算,距离乘以数量即可。这样就计算出了所有与x轴平行的周长上的边的总长度。
之后让扫描线与x轴平行即可计算出与y轴平行的所有周长上的边的总长度。
#include <cstdio>
#include <algorithm>
using namespace std;
//scanning from left to right
//discretionize Ys
#define MAX_REC_NUM 5005
#define MAX_INTERVAL MAX_REC_NUM * 2 struct Node
{
int l, r;
Node *pleft, *pright;
int num;
bool to_left, to_right;
int edge_num;
}; int node_cnt;
Node tree[MAX_INTERVAL * ]; struct Interval
{
int start, end;
int pos;
int value;
Interval()
{}
Interval(int start, int end, int pos, int value):start(start), end(end), pos(pos), value(value)
{}
bool operator < (const Interval &a)const
{
if (pos != a.pos)
return pos < a.pos;
return value > a.value;
}
}interval[MAX_REC_NUM * ]; struct Rectangle
{
int l, d, u, r;
}rec[MAX_REC_NUM]; int discrete[MAX_REC_NUM * ];
int discrete_num;
int rec_num;
int interval_num; int get_index(int a)
{
return lower_bound(discrete, discrete + discrete_num, a) - discrete;
} void discretization(int discrete[], int &discrete_num)
{
sort(discrete, discrete + discrete_num);
discrete_num = unique(discrete, discrete + discrete_num) - discrete;
} void input()
{
scanf("%d", &rec_num);
for (int i = ; i < rec_num; i++)
{
int l, d, r, u;
scanf("%d%d%d%d", &l, &d, &r, &u);
rec[i].l = l;
rec[i].r = r;
rec[i].u = u;
rec[i].d = d;
}
} void make_xscan()
{
discrete_num = ;
interval_num = ;
for (int i = ; i < rec_num; i++)
{
int l, d, r, u;
l = rec[i].l;
r = rec[i].r;
u = rec[i].u;
d = rec[i].d;
interval[interval_num++] = Interval(d, u, l, );
interval[interval_num++] = Interval(d, u, r, -);
discrete[discrete_num++] = u;
discrete[discrete_num++] = d;
}
} void make_yscan()
{
discrete_num = ;
interval_num = ;
for (int i = ; i < rec_num; i++)
{
int l, d, r, u;
l = rec[i].l;
r = rec[i].r;
u = rec[i].u;
d = rec[i].d;
interval[interval_num++] = Interval(l, r, d, );
interval[interval_num++] = Interval(l, r, u, -);
discrete[discrete_num++] = l;
discrete[discrete_num++] = r;
}
} void buildtree(Node *proot, int s, int e)
{
proot->l = s;
proot->r = e;
proot->to_left = false;
proot->to_right = false;
proot->num = ;
proot->edge_num = ;
if (s == e)
{
proot->pleft = proot->pright = NULL;
return;
}
node_cnt++;
proot->pleft = tree + node_cnt;
node_cnt++;
proot->pright = tree + node_cnt;
buildtree(proot->pleft, s, (s + e) / );
buildtree(proot->pright, (s + e) / + , e);
} void recount(Node *p)
{
if (p->num > )
{
p->edge_num = ;
p->to_right = p->to_left = true;
return;
}
if (p->pleft == NULL || p->pright == NULL)
{
p->edge_num = ;
p->to_right = p->to_left = false;
return;
}
p->to_left = p->pleft->to_left;
p->to_right = p->pright->to_right;
p->edge_num = p->pleft->edge_num + p->pright->edge_num;
if (p->pleft->to_right != p->pright->to_left)
p->edge_num++;
} void insert(Node *proot, int s, int e, int value)
{
if (s > proot->r || e < proot->l)
return;
s = max(s, proot->l);
e = min(e, proot->r);
if (s == proot->l && e == proot->r)
{
proot->num += value;
recount(proot);
return;
}
insert(proot->pleft, s, e, value);
insert(proot->pright, s, e, value);
recount(proot);
} long long work()
{
long long ans = ;
for (int i = ; i < interval_num; i++)
{
int s = get_index(interval[i].start);
int e = get_index(interval[i].end) - ;
insert(tree, s, e, interval[i].value);
long long line_num = tree->edge_num;
if (tree->to_left)
line_num++;
if (tree->to_right)
line_num++;
if (i != interval_num - )
ans += (interval[i + ].pos - interval[i].pos) * line_num;
}
return ans;
} int main()
{
input();
long long ans = ;
make_xscan();
sort(interval, interval + interval_num);
discretization(discrete, discrete_num);
buildtree(tree, , discrete_num);
ans += work(); make_yscan();
sort(interval, interval + interval_num);
discretization(discrete, discrete_num);
buildtree(tree, , discrete_num);
ans += work(); printf("%lld\n", ans);
return ;
}
poj1177的更多相关文章
- 【poj1177】 Picture
http://poj.org/problem?id=1177 (题目链接) 题意 求矩形周长并. Solution 转自:http://www.cnblogs.com/Booble/archive/2 ...
- [POJ1177]Picture
[POJ1177]Picture 试题描述 A number of rectangular posters, photographs and other pictures of the same sh ...
- 【HDOJ1828&&POJ1177】Picture(线段树,扫描线)
题意:给定n个矩形,求他们的并的周长 n<=5e3,abs(x[i])<=1e4 思路:From https://www.cnblogs.com/kuangbin/archive/2013 ...
- POJ1177 Picture —— 求矩形并的周长 线段树 + 扫描线 + 离散化
题目链接:https://vjudge.net/problem/POJ-1177 A number of rectangular posters, photographs and other pict ...
- 扫描线矩形周长的并 POJ1177
//扫描线矩形周长的并 POJ1177 // 我是按x轴 #include <iostream> #include <cstdio> #include <cstdlib& ...
- POJ1177 Picture 线段树+离散化+扫描线
求最终的覆盖图形周长,写这种代码应该短而精确,差的比较远 /* Problem: 1177 User: 96655 Memory: 348K Time: 32MS Language: C++ Resu ...
- POJ1177+线段树+扫描线
思路: 以y的值进行离散化 根据x的值 对每一条y轴边进行处理,如果是"左边"则插入,是"右边"则删除. /* 扫描线+线段树+离散化 求多个矩形的周长 */ ...
- POJ-1177 Picture 矩形覆盖周长并
题目链接:http://poj.org/problem?id=1177 比矩形面积并麻烦点,需要更新竖边的条数(平行于x轴扫描)..求横边的时候,保存上一个结果,加上当前长度与上一个结果差的绝对值就行 ...
- IOI1998 hdu1828 poj1177 Picture
写了一发扫描线竟然狂WA不止,hdu死活过不了,poj和当时IOI的数据(还花了我1dsdn积分..)都过了. 然后看到谋篇blog里有评论,把数据拿下来发现WA了. 数据是 20 0 1 11 0 ...
随机推荐
- Yii2创建多界面主题(Theme)
Yii2界面主题上的设计总体上和Yii1.x保持一致,区别在于两个地方: 1. 由于Yii2引入了独立的视图(View)类,因此界面主题(Theme)也交由视图来管理: 2. 视图文件和Web资源在目 ...
- mysql主从复制(超简单)
mysql主从复制(超简单) 怎么安装mysql数据库,这里不说了,只说它的主从复制,步骤如下: 1.主从服务器分别作以下操作: 1.1.版本一致 1.2.初始化表,并在后台启动mysql ...
- Linux下vim查看文件名
在vim下编辑时,有时候看不到文件名,不知道编辑的是那个文件,怎么呢,可以按照下面的方法试试. 查看文件名 在正常模式下: :f 或CTRL+G 查看文件的路径 用 :!pwd 可以看当前的详细路径. ...
- 成功的背后!(给所有IT人)
转载:来自CSDN第一名博主:http://blog.csdn.net/phphot/article/details/2187505 成功的背后,有着许多不为人知的故事,而正是这些夹杂着泪水和汗水的过 ...
- Slave_SQL_Running: No mysql同步故障解决方法
Slave_SQL_Running: No mysql同步故障解决 今天检查数据库发现一台MySQL Slave未和主机同步,查看Slave状态:mysql> show slave s ...
- 懒加载的用处和赋nil操作[iOS开发教程]
懒加载的用处和赋nil操作 1:数据,清空操作: self.array = nil; 2:归档从新从本地获取数据 self.archive = nil; ##id = nil的用处 block当参数, ...
- UI第一节—— UILable
1.首先说说怎么创建UI程序,打开xcode,选择Create a new Xcode project.看如下截图 2,接下来就蹦出一个和写OC应用差不多的界面,不多解释了 3.我给工程取得名字就叫 ...
- Mac 命令
1.du 获取某个目录下各个文件和子目录占用多少空间,可以输入:du -sh *
- InnoDB锁机制分析
InnoDB锁机制常常困扰大家,不同的条件下往往表现出不同的锁竞争,在实际工作中经常要分析各种锁超时.死锁的问题.本文通过不同条件下的实验,利用InnoDB系统给出的各种信息,分析了锁的工作机制.通过 ...
- Javascript高级程序设计——执行环境与作用域
Javascript中执行环境是定义了变量或函数有权访问的其他数据,决定了各自的行为,每个执行的环境都有一个与之关联的变量对象,环境中定义的所以变量和函数都保存在这个对象中. 全局执行环境是最外围的一 ...