HDU 3642 - Get The Treasury - [加强版扫描线+线段树]
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3642
Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)
Jack can’t get the total volume of the treasury because these regions don’t always contain treasury. Through years of experience, he discovers that if a region is detected that may have treasury at more than two different spots, the region really exist treasure. And now Jack only wants to know the minimum volume of the treasury.
Now Jack entrusts the problem to you.
Each test case is given in some lines. In the first line there is an integer n (1 ≤ n ≤ 1000), the number of spots on the surface of the earth that he had detected. Then n lines follow, every line contains six integers x1, y1, z1, x2, y2 and z2, separated by a space. The absolute value of x and y coordinates of the vertices is no more than 106, and that of z coordinate is no more than 500.
题意:
给定一些长方体,求这些长方体相交至少3次的体积。
题解:
z坐标的范围 [-500, 500],比较小,所以可以枚举 z平面,用类似于扫描线扫描二维图形的方式,用扫描面扫描整个三维图形,而这些扫描面,就是所有长方体的上下平面,
那么,每个我们扫描得到的截面,就可以按照HDU1542里那样的普通的二维扫描线+线段树来做,求出这样一个截面上,重叠三次及以上的面积有多大,
同时,不难想到,若我们把长方体竖直的四个侧面,看成“下闭上开”(下沿取得到,上沿取不到)的话,当我们枚举到某一个平面 $z_i$ 时,该平面截得的截面,在区间 $\left[ {z_i ,z_{i + 1} } \right)$ 都是不会变动的,这个和普通的二维扫描线+线段树的道理是一样的。
故,枚举到某一个平面 $z_i$ 时,该平面截得的截面积,乘上高度 ${z_{i + 1} - z_i }$ 之后,就是整个“3+次重叠体”在三维区间 $\left[ {z_i ,z_{i + 1} } \right)$ 内的体积。
AC代码:
#include<bits/stdc++.h>
typedef long long ll;
using namespace std; const int maxn=;
const int maxz=; int n; vector<int> x;
inline int getidx(int val){return lower_bound(x.begin(),x.end(),val)-x.begin();} vector<int> z; int tot;
struct Segment
{
int x1,x2,y;
int z1,z2;
int flag;
bool operator <(const Segment &oth)const{
return y<oth.y;
}
}segment[*maxn],tmp[*maxn]; /********************************* Segment Tree - st *********************************/
struct Node{
int l,r;
int s;
int once,twice,more;
void show()
{
printf("l=%d r=%d s=%d once=%d twice=%d more=%d\n",l,r,s,once,twice,more);
}
}node[*maxn];
void pushup(int rt)
{
int ls=rt*,rs=rt*+;
if(node[rt].s>)
{
node[rt].more=x[node[rt].r+]-x[node[rt].l];
node[rt].once=node[rt].twice=;
}
else if(node[rt].s==)
{
if(node[rt].l==node[rt].r)
{
node[rt].more=;
node[rt].twice=x[node[rt].r+]-x[node[rt].l];
node[rt].once=;
}
else
{
node[rt].more=node[ls].once+node[ls].twice+node[ls].more+node[rs].once+node[rs].twice+node[rs].more;
node[rt].twice=x[node[rt].r+]-x[node[rt].l]-node[rt].more;
node[rt].once=;
}
}
else if(node[rt].s==)
{
if(node[rt].l==node[rt].r)
{
node[rt].more=;
node[rt].twice=;
node[rt].once=x[node[rt].r+]-x[node[rt].l];
}
else
{
node[rt].more=node[ls].twice+node[ls].more+node[rs].twice+node[rs].more;
node[rt].twice=node[ls].once+node[rs].once;
node[rt].once=x[node[rt].r+]-x[node[rt].l]-node[rt].more-node[rt].twice;
}
}
else
{
if(node[rt].l==node[rt].r)
{
node[rt].more=;
node[rt].twice=;
node[rt].once=;
}
else
{
node[rt].more=node[ls].more+node[rs].more;
node[rt].twice=node[ls].twice+node[rs].twice;
node[rt].once=node[ls].once+node[rs].once;
}
}
//printf("now pushup rt=%d\t",rt); node[rt].show();
}
void build(int rt,int l,int r)
{
if(l>r) return;
node[rt].l=l; node[rt].r=r;
node[rt].s=;
node[rt].once=node[rt].twice=node[rt].more=;
if(l==r) return;
else
{
int mid=l+(r-l)/;
build(rt*,l,mid);
build(rt*+,mid+,r);
pushup(rt);
}
}
void update(int root,int st,int ed,int val)
{
if(st>node[root].r || ed<node[root].l) return;
if(st<=node[root].l && node[root].r<=ed)
{
node[root].s+=val;
pushup(root);
}
else
{
update(root*,st,ed,val);
update(root*+,st,ed,val);
pushup(root);
}
}
/********************************* Segment Tree - st *********************************/ int main()
{
int T;
scanf("%d",&T);
for(int kase=;kase<=T;kase++)
{
scanf("%d",&n); x.clear();
z.clear();
tot=;
for(int i=;i<=n;i++)
{
int x1,y1,z1,x2,y2,z2;
scanf("%d%d%d%d%d%d",&x1,&y1,&z1,&x2,&y2,&z2); segment[tot].x1=x1; segment[tot].x2=x2; segment[tot].y=y1;
segment[tot].z1=z1; segment[tot].z2=z2;
segment[tot].flag=;
tot++; segment[tot].x1=x1; segment[tot].x2=x2; segment[tot].y=y2;
segment[tot].z1=z1; segment[tot].z2=z2;
segment[tot].flag=-;
tot++; x.push_back(x1);
x.push_back(x2);
z.push_back(z1);
z.push_back(z2);
} sort(x.begin(),x.end());
x.erase(unique(x.begin(),x.end()),x.end()); sort(z.begin(),z.end());
z.erase(unique(z.begin(),z.end()),z.end()); ll ans=;
for(int i=;i<z.size()-;i++)
{
//printf("now z=%d\n",z[i]);
int cnt=;
for(int j=;j<tot;j++) if(segment[j].z1<=z[i] && segment[j].z2>z[i]) tmp[cnt++]=segment[j];
sort(tmp,tmp+cnt); build(,,x.size());
ll area=;
for(int j=;j<cnt-;j++)
{
int l=getidx(tmp[j].x1);
int r=getidx(tmp[j].x2);
//printf("now update y=%d [%d,%d](%d,%d) += %d\n",tmp[j].y,x[l],x[r],l,r-1,tmp[j].flag);
update(,l,r-,tmp[j].flag);
area+=(ll)node[].more*(tmp[j+].y-tmp[j].y);
}
ans+=area*(z[i+]-z[i]);
} printf("Case %d: %I64d\n",kase,ans);
}
}
HDU 3642 - Get The Treasury - [加强版扫描线+线段树]的更多相关文章
- HDU 3265/POJ 3832 Posters(扫描线+线段树)(2009 Asia Ningbo Regional)
Description Ted has a new house with a huge window. In this big summer, Ted decides to decorate the ...
- hdu 4419 Colourful Rectangle (离散化扫描线线段树)
Problem - 4419 题意不难,红绿蓝三种颜色覆盖在平面上,不同颜色的区域相交会产生新的颜色,求每一种颜色的面积大小. 比较明显,这题要从矩形面积并的方向出发.如果做过矩形面积并的题,用线段树 ...
- HDU 1255 覆盖的面积 (扫描线 线段树 离散化 矩形面积并)
题目链接 题意:中文题意. 分析:纯手敲,与上一道题目很相似,但是刚开始我以为只是把cnt>=0改成cnt>=2就行了,. 但是后来发现当当前加入的线段的范围之前 还有线段的时候就不行了, ...
- hdu 3642 Get The Treasury(扫描线)
pid=3642" style="">题目链接:hdu 3642 Get The Treasury 题目大意:三维坐标系,给定若干的长方体,问说有多少位置被覆盖3次 ...
- hdu 3642 Get The Treasury
Get The Treasury http://acm.hdu.edu.cn/showproblem.php?pid=3642 Time Limit: 10000/5000 MS (Java/Othe ...
- 【BZOJ3958】[WF2011]Mummy Madness 二分+扫描线+线段树
[BZOJ3958][WF2011]Mummy Madness Description 在2011年ACM-ICPC World Finals上的一次游览中,你碰到了一个埃及古墓. 不幸的是,你打开了 ...
- 【bzoj4491】我也不知道题目名字是什么 离线扫描线+线段树
题目描述 给定一个序列A[i],每次询问l,r,求[l,r]内最长子串,使得该子串为不上升子串或不下降子串 输入 第一行n,表示A数组有多少元素接下来一行为n个整数A[i]接下来一个整数Q,表示询问数 ...
- hdu1542 Atlantis(扫描线+线段树+离散)矩形相交面积
题目链接:点击打开链接 题目描写叙述:给定一些矩形,求这些矩形的总面积.假设有重叠.仅仅算一次 解题思路:扫描线+线段树+离散(代码从上往下扫描) 代码: #include<cstdio> ...
- P3722 [AH2017/HNOI2017]影魔(单调栈+扫描线+线段树)
题面传送门 首先我们把这两个贡献翻译成人话: 区间 \([l,r]\) 产生 \(p_1\) 的贡献当且仅当 \(a_l,a_r\) 分别为区间 \([l,r]\) 的最大值和次大值. 区间 \([l ...
随机推荐
- 使用Socket抓取网页源码
import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; import ...
- Lambda模式
Class TestPatterns '<<1.Self-Defining Functions-自定义函数>> Public Shared Sub SelfDefining() ...
- my-small.ini、my-medium.ini、my-large.ini、my-huge.ini文件的作用
安装完mysql之后或者是下载的免安装版解压之后,默认是没有my.ini文件的.但是,有几个类似的文件,如my-small.ini.my-medium.ini.my-large.ini.my-huge ...
- NetBpm 示例:请假审批(6)
转载注明出处: http://www.cnblogs.com/anbylau2130/p/3877983.html 原文: 请假示例 流程定义包源码下载(注:par包就是zip格式压缩包).原文地址: ...
- 入门常用SQL及官方文档的使用
SQL语句基础理论 SQL是操作和检索关系型数据库的标准语言,标准SQL语句可用于操作关系型数据库. 5大主要类型: ①DQL(Data Query Language,数据查询语言)语句,主要由于se ...
- Oracle和Mysql中mybatis模糊匹配查询区别
1.Oracle AND NAME LIKE '%'||#{name}||'%' 2.Mysql AND NAME LIKE "%"#{name}"%"
- JS 数组Array常用方法
参考网站: http://www.jb51.net/article/60502.htm,作者:junjie 今天在使用js切割字符串"浙江,江苏 , 天津,"...这样字符串的时候 ...
- 【技术分享会】 @第六期 iOS开发基础
前言 iOS之前被称为 iPhone OS,是一个由苹果公司开发的移动操作系统. iOS的第一个版本是在2007年发布的,其中包括iPhone和iPod Touch. iOS开发工具:Xcode 运行 ...
- Android设计和开发系列第二篇:Action Bar(Develop—Training)
Adding the Action Bar GET STARTED DEPENDENCIES AND PREREQUISITES Android 2.1 or higher YOU SHOULD AL ...
- Qt封装百度人脸识别+图像识别
AI技术的发展在最近几年如火如荼,工资待遇也是水涨船高,应用的前景也是非常广阔,去年火起来的人脸识别,今年全国遍地开花,之前封装了下face++的人脸识别等接口,今年看了下百度的AI,还免费了,效果也 ...