Description

二维平面中,给定   N个等腰直角三角形(每个三角形的两条直角边分别
 
 
平行于坐标轴,斜边从左上到右下)。我们用三个非负整数( xyd)来描
 
述这样一个三角形,三角形三个顶点的坐标
 
分别为(xy), (x + dy)和(xy +     d)。要求计算这   N个三角形所覆盖的总面
 
积。例如,下图有 3 个三角形,覆盖的总面积为 11.0。
 

Input

输入文件第一行为一个正整数N,表示三角形的个数。接下来的 N

行每行有用空格隔开的三个非负整数,  xy   , d,描述一个三角

形的顶点坐标,分别为

(    xy), (x + dy), (  xy+d),

其中 xyd 满足0≤   xyd≤1000000。

对于50%的数据,1≤         N≤500;

100%的数据,1≤N≤10000。

Output

仅包含一行,为一个实数    S    ,表示所有三角形所覆盖的总面积,输出恰

好保留一位小数。输入数据保证     S≤2^31 。

Sample Input

3 .
1 1 4
2 0 2
3 2 2

Sample Output

11.0

题解:

将所有三角形按照底边的y坐标升序排序,然后用一根扫描线,从最下面的三角形的底边开始向上扫,并时刻维护sum[]数组和len,sum[i]=第i格上覆盖的三角形个数(注意是第i格而不是第i个点!第i个点和第i+1个点之间的那一格就是第i格),len=扫描线上被覆盖的部分的总长度(以下简称有效长度),显然移动完一次扫描线后,答案中增加的面积=$$\frac{上次扫描线的有效长度+本次扫描线的有效长度}{2}$$

那么实际上整个题要做的就是从最下面不断地向上移动扫描线,用一个栈维护覆盖在扫描线上面的三角形,每移动一次扫描线,先维护一次有效长度和sum[]数组,但是这次维护只是在原有的三角形基础上减少,并不增加三角形(也就是说这次维护是不添加新相交的三角形的,有效长度只会减少,不会增加)!在答案中添加面积后,再对扫描线上的有效长度和sum[]数组这两个信息进行第二次维护,这次会加入移动扫描线后新相交的三角形。如此反复便可得到答案。

但是这样做还是会TLE掉最后两个点,因此我们需要再想办法优化,可以发现,某些小三角形是被大三角形所包裹起来的(如下图中红叉的那个紫色三角形,虚线是扫描线,箭头是扫描线的移动方向),显然这样的小三角形都可以忽略不计,因此可以大大减少数据规模。

那么我们可以标记每个三角形是否已经被删除了,在每次扫描线移动后,在加入新的三角形的时候,看这个三角形是否包裹了原来的三角形,以及这个三角形是否被原来的三角形所包裹。若这个三角形包裹了原来的三角形,就把原来的那个三角形删掉,如此下去,如果这个三角形并没有被原来扫描线上的三角形包裹,那么就把它加入扫描线上,并更新对应于扫描线上的区间的格子的信息。 
要做到轻松地删除三角形,并通过这样的优化减少数据规模的话,就需要用一个双向链表来维护当前所有还没被删掉的三角形,当然也可以用splay来维护的啦,速度会快很多。

 //Never forget why you start
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
using namespace std;
int n,m;
struct triangle {
int x,y,d,l,r;//横坐标,纵坐标,直角边长,左端点横坐标,右端点横坐标
friend bool operator > (const triangle a,const triangle b) {
return a.l<=b.l&&b.r<=a.r;
}//定义一个三角形a左右两端点如果完全包含另外一个三角形b,则a>b
} a[];
int Next[],pre[],vis[],head,mmax,mmin,len,lastlen,sum[],top,stack[];
//Next[]表示双向链表的下一个数
//pre[]表示双向链表的上一个数
//vis[]表示是否在双向链表中
//head表示双向链表的第一个点
double ans;
void delet(int x) {
vis[x]=;
if(head==x)head=Next[x];
pre[Next[x]]=pre[x];
Next[pre[x]]=Next[x];
}//从双向链表中删除一个数
bool cmp(triangle a,triangle b) {
if(a.y==b.y)return a.l<=b.l&&b.r<=a.r;
else return a.y<b.y;
}//比较函数,先按y从小到大排序,再使前面的三角形完全包含后面的三角形
int main() {
int i,j,k,l;
scanf("%d",&n);
for(i=; i<=n; i++) {
scanf("%d%d%d",&a[i].x,&a[i].y,&a[i].d);
a[i].r=a[i].x+a[i].d;
a[i].l=a[i].x;
vis[i]=;
Next[i]=i+;
pre[i]=i-;
mmin=min(mmin,a[i].y);
mmax=max(mmax,a[i].y+a[i].d);
}//输入,初始化双向链表
sort(a+,a+n+,cmp);//排序
head=;
for(i=head; i<=n&&a[i].y==mmin; i=Next[i]) {
for(j=a[i].l; j<a[i].r; j++) {
if(!sum[j])len++;
sum[j]++;
}
}//统计最开始被三角形覆盖的长度
for(i=mmin+; i<=mmax; i++) {
lastlen=len;//lastlen表示层被覆盖的长度,len表示这一层备覆盖的长度
top=;//初始化栈
for(j=head; j<=n&&a[j].y<i; j=Next[j]) {//遍历上一层已经计算过的三角形
a[j].r--;//因为向上挪了一层,就相当于宽度减小了1
if(a[j].r<a[j].l)delet(j);//如果这个三角形完全消失,则从双向链表中删去
else {
if(sum[a[j].r]==)len--;//判断r--后是否对len的大小造成影响
sum[a[j].r]--;
stack[++top]=j;//将这个三角形压入栈中(这个三角形覆盖了一部分长度)
}
}
ans+=(double)(lastlen+len)/2.0;//计算答案
for(j=head; j<=n&&a[j].y<=i; j=Next[j]) {
if(a[j].y==i) {//遍历所有覆盖在当前层的三角形
for(k=; k<=top; k++) {//遍历所有在栈中的三角形,
if(vis[stack[k]]==)continue;
if(a[stack[k]]>a[j]) {//判断当前三角形是非已被栈中某个三角形覆盖
delet(j);//如果已经覆盖,就在双向链表中将这个三角形删除
break;
}
if(a[j]>a[stack[k]]) {
delet(stack[k]);
for(l=a[stack[k]].l; l<a[stack[k]].r; l++) {
if(sum[l]==)len--;
sum[l]--;
}//如果栈中三角形被当前三角形覆盖,就删除栈中三角形
}
}
if(k==top+) {//如果所有栈中三角形都无法将当前三角形覆盖,就加入当前三角形
for(l=a[j].l; l<a[j].r; l++) {
if(!sum[l])len++;
sum[l]++;
}
}
}
}
}
printf("%.1lf\n",ans);
return ;
}

[HNOI 2012]三角形覆盖问题的更多相关文章

  1. 【BZOJ2731】三角形覆盖问题

    想象一条平行于\(y\)轴的扫描线,从低往高扫描.如何确定关键高度才能使每两个关键高度之间分割出的图形易于计算呢? 关键高度有:三角形底边高度.三角形上顶点高度.三角形交点的高度. ​ 如此分割,我们 ...

  2. [HNOI2012]三角形覆盖问题

    题面 二维平面中,给定 \(N\) 个等腰直角三角形(每个三角形的两条直角边分别平行于坐标轴,斜边从左上到右下).我们用三个非负整数 \((x, y, d)\) 来描述这样一个三角形,三角形三个顶点的 ...

  3. 【题解】三角形 [P1222] / 三角形覆盖问题 [HNOI2012] [P3219]

    [题解]三角形 [P1222] / 三角形覆盖问题 [HNOI2012] [P3219] 传送门: 三角形 \(\text{[P1222]}\) 三角形覆盖问题 \(\text{[HNOI2012] ...

  4. BZOJ 2731 Luogu P3219 [HNOI2012]三角形覆盖问题 (扫描线)

    题目链接: (bzoj)https://www.lydsy.com/JudgeOnline/problem.php?id=2731 (luogu)https://www.luogu.org/probl ...

  5. HNOI 2012 永无乡

    codevs 1477 永无乡 http://codevs.cn/problem/1477/ 2012年湖南湖北省队选拔赛  时间限制: 1 s  空间限制: 128000 KB   题目描述 Des ...

  6. 【BZOJ 2733】【HNOI 2012】永无乡 Splay启发式合并

    启发式合并而已啦,, 调试时发现的错误点:insert后没有splay,把要拆开的树的点插入另一个树时没有把ch[2]和fa设为null,找第k大时没有先减k,,, 都是常犯的错误,比赛时再这么粗心就 ...

  7. BZOJ 2733 HNOI 2012 永无乡 平衡树启示式合并

    题目大意:有一些岛屿,一開始由一些无向边连接. 后来也有不断的无向边增加,每个岛屿有个一独一无二的重要度,问随意时刻的与一个岛屿联通的全部岛中重要度第k大的岛的编号是什么. 思路:首先连通性一定要用并 ...

  8. HNOI 2012 矿场搭建

    #include<iostream> #include<cstdio> #include<cstdlib> #include<cstring> #def ...

  9. [HNOI 2012]集合选数

    Description 题库链接 对于任意一个正整数 \(n\) ,求出集合 \(\{1,2,\cdots,n\}\) 的满足约束条件"若 \(x\) 在该子集中,则 \(2x\) 和 \( ...

随机推荐

  1. Codeforces Round #397 题解

    Problem A. Neverending competitions 题目大意 一个团队有多个比赛,每次去比赛都会先订机票去比赛地点,然后再订机票返回.给出\(n\)个含有起止地点的购票记录(不按时 ...

  2. tyvj 1203 机器分配

    时间: 1000ms / 空间: 131072KiB / Java类名: Main 描述 总公司拥有高效生产设备M台,准备分给下属的N个公司.各分公司若获得这些设备,可以为国家提供一定的盈利.问:如何 ...

  3. vue文件名规范

    之前有看过一些命名规范,也看到说vue文件命名要么全是小写要么就是用小写 + '-':其实看到的时候有点不以意,因为本地能跑起项目:发布能正常访问也就OK了. 但是今天在做自动化部署的时候碰到一个问题 ...

  4. BZOJ4571:[SCOI2016]美味

    浅谈主席树:https://www.cnblogs.com/AKMer/p/9956734.html 题目传送门:https://www.lydsy.com/JudgeOnline/problem.p ...

  5. .Net 学习资源整理

    01.Visual Studio 隐藏的财富 --- C# 语言规范 安装完Visual Studio之后,我们好像忽略了,微软给我们准备的<C# 语言规范>. 路径参考下图: 02.MS ...

  6. 'ascii' codec can't decode byte 0xe6 in position 0: ordinal not in range(128)

    No valid host was found. There are not enough hosts available 'ascii' codec can't decode byte 0xe6 i ...

  7. 如何将Eclipse中编写的java项目导出?

    转自:https://zhidao.baidu.com/question/347808396.html1.导入项目 当下载了包含Eclipse 项目的源代码文件后,我们可以把它导入到当前的Eclips ...

  8. maven 依赖范围

  9. 在Spring环境下存取properties文件…

    Spring中PropertyPlaceholderConfigurer的使用 (1) 基本的使用方法是 classpath:/spring/include/dbQuery.properties 其中 ...

  10. 一个基于Tp3.2(thinkphp3.2)的工会管理系统

    该系统包括11个模块. 会员管理模块 奖惩管理模块 运动会管理模块 新闻管理模块 文档管理模块 经费管理模块 电子提案管理模块 用户管理模块 权限管理模块 系统管理模块 系统的登录 系统主页 这里只是 ...