UVA 12171 (hdu 2771)sculptrue(离散化)
以前对离散化的理解不够,所以把端点和区间区分来考虑但是做完这题以后有了新的认识:
先来看一个问题:给你以下的网格,你需要多少空间去存储红点区间的信息呢?

只需要图上所示的1,2,3,4个点就足够表示红点所在区间了,为什么不是一个区间的第一个红点和最后一个红点呢?(如果这样记录的话则必须加一区间点,记录区间内部信息,因为端点可能是两个区间的交集而区间内可能只被操作了一次)这样做的好处是空白区域的长度也能轻易计算出来。
因此离散化的核心在于以点代表区间。

为了计算总区间两端空白的长度,增加A和B点。
原数据找离散后的值直接二分,没必要建map,因为map的查询也是logN级别的
知道了这些,存储问题就解决了。
/*
Created by Rey Chen on 2015.6.25
*/
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<queue>
//#define local
using namespace std;
const int maxn = + ;
const int max2n = + ;
const int maxc = + ;//coordinate;
typedef int ll;
// original data
int n, x0[maxn], y0[maxn], z0[maxn], x1[maxn], y1[maxn], z1[maxn];
// discretized data
int xs[max2n], ys[max2n], zs[max2n];
int xn, yn, zn;//the numbers of each coordinate //1 means solid, 2 means visited air
int color[max2n][max2n][max2n]; void discretize(int* x, int& n) {
sort(x, x+n);
n = unique(x, x+n) - x;
} struct Cell
{
int x, y, z;
Cell(int x,int y,int z):x(x),y(y),z(z){}
};
//related to flood fill
const int dx[] = {,-,,,,};
const int dy[] = {,,,-,,};
const int dz[] = {,,,,,-}; #define vis(u) (color[v.x][v.y][v.z] == 2)
#define setVis(u) color[u.x][u.y][u.z] = 2//solid..
#define uNeigh(i) Cell v(u.x+dx[i], u.y+dy[i], u.z+dz[i])
#define inside(v) (v.x >=0 && v.x < xn && v.y >= 0 && v.y < yn && v.z >= 0 && v.z < zn)
#define issolid(v) (color[v.x][v.y][v.z] == 1)
#define dis(u,x) (x##s[u.x+1]-x##s[u.x])
#define vol(u) (ll)(xs[u.x+1]-xs[u.x])*(ys[u.y+1]-ys[u.y])*(zs[u.z+1]-zs[u.z])//为了让原本最后一层空气计算时不下标不越界 还要再加一层 inline ll area(Cell& v, int dir){
if(dx[dir]) return (ll)dis(v,y)*dis(v,z);
if(dy[dir]) return (ll)dis(v,x)*dis(v,z);
return (ll)dis(v,x)*dis(v,y);
} //dfs might stack overflow
void floodfill(ll &V,ll &s)
{
queue<Cell> q;
Cell u(,,);// Cell u = Cell() 1,2
setVis(u);
q.push(u);
V = s = ;
while(!q.empty()) {
u = q.front();q.pop();
V += vol(u); //calculate air volume
for(int i = ; i < ; i++) {
uNeigh(i);
if(!inside(v)) continue;//最后一层不加
if(issolid(v)) s += area(v,i); //ensure air vis once meet solid add area
else if(!vis(v)) {
setVis(v);
q.push(v);
}
}
}
V = maxc*maxc*maxc - V;
} #define add(x) x##1[i] += x##0[i]
#define dadd(a) xs[t] = x##a[i]; ys[t] = y##a[i]; zs[t++] = z##a[i]
#define ID(x,n,val) (lower_bound(x,x+n,val) - x)
#define GetId(X,x) X##0 = ID(x##s, x##n, x##0[i]);X##1 = ID(x##s, x##n, x##1[i])
//X0 = ID(xs,xn,x0[i]);
int main()
{
#ifdef local
freopen("in.txt","r",stdin);
#endif // local int T;
scanf("%d",&T);
while(T--) {
scanf("%d",&n); int t = ;
*xs = *ys = *zs = ;//add air
xs[] = ys[] = zs[] = maxc;//最后面有两层空气
for(int i = ; i < n; i++) {
scanf("%d%d%d%d%d%d", x0+i, y0+i, z0+i, x1+i, y1+i, z1+i);
add(x);add(y);add(z);
dadd(); dadd();
}
xn = yn = zn = t ;
discretize(xs,xn);
discretize(ys,yn);
discretize(zs,zn);
xn--;yn--;zn--;//不访问最后一层空气,计算总体积时要加上最后一层空气 memset(color,,sizeof(color));
for(int i = ; i < n; i++) { //solid set
int X0, X1, Y0, Y1, Z0, Z1;
GetId(X,x); GetId(Y,y); GetId(Z,z);
for(int X = X0; X < X1; X++)
for(int Y = Y0; Y < Y1; Y++)
for(int Z = Z0; Z < Z1; Z++)
color[X][Y][Z] = ;
}
ll v,s;
floodfill(v,s);
printf("%d %d\n",s,v);
}
return ;
}
UVA 12171 (hdu 2771)sculptrue(离散化)的更多相关文章
- uva 12171 hdu 1771 Sculpture
//这题从十一点开始写了四十分钟 然后查错一小时+ 要吐了 这题题意是给很多矩形的左下角(x,y,z最小的那个角)和三边的长(不是x,y,z最大的那个角T-T),为组成图形的面积与表面积(包在内部的之 ...
- hdu 2771(uva 12171) Sculpture bfs+离散化
题意: 给出一些边平行于坐标轴的长方体,这些长方体可能相交.也可能相互嵌套.这些长方体形成了一个雕塑,求这个雕塑的整体积和表面积. 题解: 最easy想到直接进行bfs或者dfs统计,但此题的麻烦之处 ...
- Uva 12171 Sculpture - 离散化 + floodfill
题目连接:https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem ...
- UVa 12171 (离散化 floodfill) Sculpture
题意: 三维空间中有n个长方体组成的雕塑,求表面积和体积. 分析: 我们可以在最外边加一圈“空气”,然后求空气的连通块的体积,最后用总体积减去即是雕塑的体积. 还有一个很“严重”的问题就是5003所占 ...
- UVa 221城市正视图(离散化)
https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem& ...
- UVA 10816 + HDU 1839 Dijstra + 二分 (待研究)
UVA 题意:两个绿洲之间是沙漠,沙漠的温度不同,告诉起点,终点,求使得从起点到终点的最高温度最小的路径,如果有多条,输出长度最短的路径: 思路:用最小费用(最短路径)最大流(最小温度)也能搞吧,但因 ...
- HDU - 1255 扫描线+离散化进阶
这道题最开始我以为和HDU - 1542 那道题一样,只需要把cover次数改成2次即可,但是后面仔细一想,我们需要求的是覆盖次数大于等于2次的,这样的话,我们需要维护两个长度,HDU-1542 由于 ...
- HDU 5925 Coconuts 离散化
题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5925 Coconuts Time Limit: 9000/4500 MS (Java/Others) ...
- hdu 5303 DP(离散化,环形)+贪心
题目无法正常粘贴,地址:http://acm.hdu.edu.cn/showproblem.php?pid=5303 大意是给出一个环形公路,和它的长度,给出若干颗果树的位置以及树上的果子个数. 起点 ...
随机推荐
- 2-7 Flutter开发工具使用指南
这里选择用哪个模拟器运行 Mac系统下可以通过这个Open IOS Siumlator打开IOS模拟器 debug用来调试的 可以创建新的模拟器 选择安卓模拟器的版本 这是sdk的配置 点开就是打开了 ...
- SpringBoot使用拦截器无效
附上代码: public class WendaWebConfiguration extends WebMvcConfigurerAdapter { @Autowired PassportInterc ...
- Matcher的group()/group(int group)/groupCount()用法介绍
直接上代码: package com.dajiangtai.djt_spider.util; import java.util.regex.Matcher;import java.util.regex ...
- 洛谷P3379 【模板】最近公共祖先(LCA)(dfs序+倍增)
P3379 [模板]最近公共祖先(LCA) 题目描述 如题,给定一棵有根多叉树,请求出指定两个点直接最近的公共祖先. 输入输出格式 输入格式: 第一行包含三个正整数N.M.S,分别表示树的结点个数.询 ...
- ansible无网络安装openstack(Newton)
概要 apt使用本地源,pip使用本地源 网络环境 物理机环境 网络名 网络地址 VLAN br-ex 192.168.200.250/24 90 br-mgmt 92.0.0.100/24 92 a ...
- 小a和uim之大逃离(luogu P1373 dp)
小a和uim之大逃离(luogu P1373 dp) 给你一个n*m的矩阵,其中元素的值在1~k内.限制只能往下和往右走,问从任意点出发,到任意点结束,且经过了偶数个元素的合法路径有多少个.在此题中, ...
- java之代理 静态代理和动态代理
一.静态代理 1. 代理有很多种,有虚拟代理,保护代理,智能引用代理,和远程代理; 开发中最常用的是只能引用代理 2. 代理的模式有两种,分别是: 静态代理 代理对象,被代理对象在 ...
- jquery jtemplates.js模板渲染引擎的详细用法第三篇
jquery jtemplates.js模板渲染引擎的详细用法第三篇 <span style="font-family:Microsoft YaHei;font-size:14px;& ...
- easyui---panel(面板)
panel笔记: EASYUI panel: class:easyui-panel,带有title 打开:onclick="javascript:$('#c').panel('open')& ...
- ES6新特性使用小结(六)
十三.promise 异步编程 ①.使用 promise 模拟异步操作 { //ES5 中的 callback 解决 异步操作问题 let ajax = function (callback) { c ...