uva oj 12171(hdu 2771)[离散化]sculptrue

以前对离散化的理解不够,所以把端点和区间区分来考虑但是做完这题以后有了新的认识:

先来看一个问题:给你以下的网格,你需要多少空间去存储红点区间的信息呢?

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

为了计算总区间两端空白的长度,增加A和B点。

原数据找离散后的值直接二分,没必要建map,因为map的查询也是logN级别的

知道了这些,存储问题就解决了。

#include<cstdio>
#include<algorithm>
#include<cstring>
#include<queue>
//#define local
using namespace std;
const int maxn = 50 + 1;
const int max2n = 100 + 2;
const int maxc = 1000 + 1;//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[] = {1,-1,0,0,0,0};
const int dy[] = {0,0,1,-1,0,0};
const int dz[] = {0,0,0,0,1,-1};

#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(0,0,0);// Cell u = Cell() 1,2
   setVis(u);
   q.push(u);
   V = s = 0;
   while(!q.empty()) {
      u = q.front();q.pop();
      V += vol(u); //calculate air volume
      for(int i = 0; i < 6; 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 = 2;
      *xs = *ys = *zs = 0;//add air air
      xs[1] = ys[1] = zs[1] = maxc;//最后面有两层空气
      for(int i = 0; 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(0); dadd(1);
      }
      xn = yn = zn = t ;
      discretize(xs,xn);
      discretize(ys,yn);
      discretize(zs,zn);
      xn--;yn--;zn--;//不访问最后一层空气,计算总体积时要加上最后一层空气

      memset(color,0,sizeof(color));
      for(int i = 0; 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] = 1;
      }
      ll v,s;
      floodfill(v,s);
      printf("%d %d\n",s,v);
   }
   return 0;
}
时间: 2024-08-05 08:07:29

uva oj 12171(hdu 2771)[离散化]sculptrue的相关文章

UVa OJ 127 - &quot;Accordian&quot; Patience (“手风琴”纸牌)

UVa OJ 127 - "Accordian" Patience ("手风琴"纸牌) Time limit: 3.000 seconds 限时:3.000秒 Problem 问题 You are to simulate the playing of games of "Accordian" patience, the rules for which are as follows: 模拟玩一个"手风琴"纸牌游戏,规则如下: D

uva oj 567 - Risk(Floyd算法)

1 /* 2 一张有20个顶点的图上. 3 依次输入每个点与哪些点直接相连. 4 并且多次询问两点间,最短需要经过几条路才能从一点到达另一点. 5 6 bfs 水过 7 */ 8 #include<iostream> 9 #include<cstring> 10 #include<vector> 11 #include<cstdio> 12 #include<queue> 13 using namespace std; 14 15 struct

hdu 4400 离散化+二分+BFS(暴搜剪枝还超时的时候可以借鉴一下)

Mines Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Total Submission(s): 1110    Accepted Submission(s): 280 Problem Description Terrorists put some mines in a crowded square recently. The police evacuate all peo

HDU 1255 离散化+扫描线覆盖的面积

覆盖的面积 Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submission(s): 3571    Accepted Submission(s): 1753 Problem Description 给定平面上若干矩形,求出被这些矩形覆盖过至少两次的区域的面积. Input 输入数据的第一行是一个正整数T(1<=T<=100),代表测试数据的数量.每个测试数据

hdu 2771(uva 12171) Sculpture bfs+离散化

题意: 给出一些边平行于坐标轴的长方体,这些长方体可能相交.也可能相互嵌套.这些长方体形成了一个雕塑,求这个雕塑的整体积和表面积. 题解: 最easy想到直接进行bfs或者dfs统计,但此题的麻烦之处在于求整个雕塑的外表面积和雕塑内部可能出现四个长方体所搭成的空心.空心不能计算到表面积中,可是计算整体积却要计入,于是直接bfs或者dfs不优点理.于是,能够想到直接统计整个雕塑外围的全部小方块.就可以非常方便地求出雕塑地表面积和体积(雕塑地整体积==整个空间地体积-外围想方块的体积),另一点就是因

uva 12171 hdu 1771 Sculpture

//这题从十一点开始写了四十分钟 然后差错一小时+ 要吐了 这题题意是给很多矩形的左下角(x,y,z最小的那个角)和三边的长(不是x,y,z最大的那个角T-T),为组成图形的面积与表面积(包在内部的之算体积不算表面积) 解法:离散化+bfs,先把范围扩大(相当于在周围加上空气),然后bfs,遇到表面积直接加入,遇到非长方体的部分也直接加入,最后用总体积减去空气的体积,这样就可以把内部的体积计算进来而不计算其表面积.因为坐标范围比较大,要先离散化. //其实我对这题一直耿耿于怀,当年没进省队多少与

HDU 2836 (离散化DP+区间优化)

Reference:http://www.cnblogs.com/wuyiqi/archive/2012/03/28/2420916.html 题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=2836 题目大意:计算序列有多少种组合,每个组合至少两个数,使得组合中相邻两个数之差不超过H,序列有重复的数.MOD 9901. 解题思路: 朴素O(n^2) 以样例1 3 7 5为例,如果没有重复的数. 设dp[i]前n个数的方案数,初始化为1. 那么fo

UVa 221城市正视图(离散化)

https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem&problem=157 题意:输入建筑物的x,y坐标,宽度,深度和高度,输出从南向北看时能看到的建筑物. 这道题需要用到离散化,将所有建筑物物的左右边界坐标存储起来,然后排序去重,那么任意两个相邻x坐标形成的区间要么是可见的,要么就是不可见的.这样只需在这个区间内任选一点(如中点),当该建筑物可见时,首先它的

hdu -4325-Flowers(离散化 线段树)

http://acm.hdu.edu.cn/showproblem.php?pid=4325: 题目意思: 给你N个花开的时间段,然后在有M个时间点,问你在这个时间点有多少花开着. 昨天刚做的一个类似的题,用线段树加离散化,然后赶紧敲,然后错八次. 最后还是没做出来. 那么思路还是线段树加离散化. 题目给的每个花开花谢的范围很大,但花的数目只有1e5,很容易就想到了离散化,然后就是用线段树去处理,找每个花开的段,最后lazy下放到 每个时间点,这样就知道每个时间点开的花数. 这题大致思路是这样的