题意:题目背景是<<我的世界>>,堆方块造房子,堆的规则是:新堆的方块必须和已有方块有重合面,而且不能往封闭空间里堆。 在三维空间中,给定一个堆的序列,判断符不符合规则。

数据范围:


思路:如果正向考虑,判断方块是否放在封闭空间很难实现。那么就逆向考虑,删除某个方块,然后用floodfill算法(其实可以想当然的用bfs实现,我就是自己实现的这个算法)。从某个点开始,格子看做障碍物,bfs找到所有可以到达的空格。如果某个格子周围没有先前被访问(bfs标记)过的点,说明一定在封闭空间中。

AC代码

#include <cstdio>
#include <cmath>
#include <cctype>
#include <bitset>
#include <algorithm>
#include <cstring>
#include <utility>
#include <string>
#include <iostream>
#include <map>
#include <set>
#include <vector>
#include <queue>
#include <stack>
using namespace std;
#pragma comment(linker, "/STACK:1024000000,1024000000")
#define eps 1e-10
#define inf 0x3f3f3f3f
#define pii pair<int, int>
typedef long long LL;
const double PI = acos(-1.0);
const int maxn = 100 + 5;
int Fre[maxn][maxn][maxn], pos[maxn][maxn][maxn];
struct node{
    int x, y, z;
    node(){}
    node(int a, int b, int c):x(a),y(b),z(c){
    }
}cube[100005];
int lx, ly, lz, rx, ry, rz;

const int dsize = 6;
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};

void init() {
    memset(pos, 0, sizeof(pos));
    memset(Fre, 0, sizeof(Fre));
    lx = ly = lz = inf;
    rx = ry = rz = 0;
}

void getBorder(int x, int y, int z) {
    rx = max(rx, x);
    ry = max(ry, y);
    rz = max(rz, z);
    lx = min(lx, x);
    ly = min(ly, y);
    lz = min(lz, z);
}

bool isIn(int x, int y, int z) {
    if(x < lx || x > rx || y < ly || y > ry || z < lz || z > rz) return false;
    return true;
}

void floodfill(int x, int y, int z) {
    queue<node>Q;
    Fre[x][y][z] = 1;
    Q.push(node(x, y, z));
    while(!Q.empty()) {
        node now = Q.front(); Q.pop();
        x = now.x, y = now.y, z = now.z;
        for(int i = 0; i < dsize; ++i) {
            int px = x + dx[i], py = y + dy[i], pz = z + dz[i];
            if(!isIn(px, py, pz) || pos[px][py][pz] || Fre[px][py][pz]) continue;
            Fre[px][py][pz] = 1;
            Q.push(node(px, py, pz));
        }
    }
}

bool adjacent(int x, int y, int z) {
    int Free = 0, adj = 0;
    for(int i = 0; i < dsize; ++i) {
        int px = x + dx[i], py = y + dy[i], pz = z + dz[i];
        if(!isIn(px, py, pz) && pz != 0) continue;
        if((pos[px][py][pz] && !Fre[px][py][pz]) || pz == 0) adj = 1;
        if(Fre[px][py][pz]) Free = 1;
    }
    return adj && Free;
}

bool Place(int n) {
    for(int i = n-1; i >= 0; --i) {
        int x = cube[i].x, y = cube[i].y, z = cube[i].z;
        if(!adjacent(x, y, z)) return false;
        floodfill(x, y, z);
    }
    return true;
}

void in(int &a) {
    char ch;
    while((ch=getchar()) < '0' || ch > '9');
    for(a = 0; ch >= '0' && ch <= '9'; ch = getchar())
        a = a*10 + ch - '0';
}

int main() {
    int T, n;
    scanf("%d", &T);
    while(T--) {
        init();
        scanf("%d", &n);
        int x, y, z;
        for(int i = 0; i < n; ++i) {
            in(x); in(y); in(z);
            getBorder(x, y, z);
            cube[i] = node(x, y, z);
            pos[x][y][z] = 1;
        }
        rx++, ry++, rz++;
        lx--, ly--, lz = 1;
        floodfill(rx, ry, rz);
        if(Place(n)) printf("Yes\n");
        else printf("No\n");
    }
    return 0;
}

如有不当之处欢迎指出!

hiho第151周 Building in Sandbox floodfill的更多相关文章

  1. hihoCoder #1291 : Building in Sandbox 逆向处理+并查集维护

    /** 题目:#1291 : Building in Sandbox 链接:https://hihocoder.com/problemset/problem/1291 题意:就是一个三维的空间里,按照 ...

  2. hiho一下21周 线段树的区间修改 离散化

    离散化 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Hi和小Ho在回国之后,重新过起了朝7晚5的学生生活,当然了,他们还是在一直学习着各种算法~ 这天小Hi和小Ho ...

  3. hiho 第116周,最大流最小割定理,求最小割集S,T

    小Hi:在上一周的Hiho一下中我们初步讲解了网络流的概念以及常规解法,小Ho你还记得内容么? 小Ho:我记得!网络流就是给定了一张图G=(V,E),以及源点s和汇点t.每一条边e(u,v)具有容量c ...

  4. hiho一下116周 网络流

    网络流二·最大流最小割定理 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Hi:在上一周的Hiho一下中我们初步讲解了网络流的概念以及常规解法,小Ho你还记得内容么? ...

  5. hiho 第119周 最大权闭合子图

    描述 周末,小Hi和小Ho所在的班级决定举行一些班级建设活动. 根据周内的调查结果,小Hi和小Ho一共列出了N项不同的活动(编号1..N),第i项活动能够产生a[i]的活跃值. 班级一共有M名学生(编 ...

  6. hiho 第六周 01背包

    简单的01背包,没有报名,这周的没有权限提交 #include<iostream> #include<memory.h> using namespace std; #defin ...

  7. hiho一下 第一周 最长回文子串

    时间限制:1000ms 单点时限:1000ms 内存限制:64MB 描述 小Hi和小Ho是一对好朋友,出生在信息化社会的他们对编程产生了莫大的兴趣,他们约定好互相帮助,在编程的学习道路上一同前进. 这 ...

  8. hiho一下20周 线段树的区间修改

    线段树的区间修改 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 对于小Ho表现出的对线段树的理解,小Hi表示挺满意的,但是满意就够了么?于是小Hi将问题改了改,又出给了 ...

  9. hiho一下18周 RMQ问题再临

    RMQ问题再临 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 终于,小Hi和小Ho踏上了回国的旅程.在飞机上,望着采购来的特产--小Hi陷入了沉思:还记得在上上周他们去 ...

随机推荐

  1. 解决Flink输出日志中时间比当前时间晚8个小时的问题

    Flink安装在CentOS7上,默认时间是UTC时间,查看Flink日志,发现输出时间比当前时间晚8个小时. 通过如下命令,调整成北京时间 cp /usr/share/zoneinfo/Asia/S ...

  2. JavaScript 函数创建思想

    //定义一个函数的步骤//1.开辟一个新的空间地址//2.把函数体里面的代码当做字符串存储到空间里面(一个函数如果只定义了,没有执行的话,这个函数没有任何意义)//3.在把我们的地址给我们的函数名fu ...

  3. thinkphp5学习(一)——thinkphp5的目录结构与开发规范

    开发规范: 目录和文件 目录使用小写+下划线: 类库.函数文件统一以.php为后缀: 类的文件名均以命名空间定义,并且命名空间的路径和类库文件所在路径一致: 类文件采用驼峰法命名(首字母大写),其它文 ...

  4. CentOS7下 Java、Tomcat、MySQL、Maven热部署

    本文介绍了CentOS7 64位下Java.Tomcat.MySQL.Maven热部署等服务器环境的搭建和调试过程. 学生服务器资源获取方法: 云+校园计划 - 腾讯云 阿里云云翼计划 github ...

  5. Intellij IDEA 15 如何同时打开多个项目

    标题:Intellij IDEA 15 如何同时打开多个项目 作者原创技术文章,转载请注明出处 我们在编程时常常需要打开多个项目,例如操作复制黏贴或者参考其他项目等等,但是编译器Intellij ID ...

  6. fiddler2请求参数乱码

    win7 1.windows按钮+R 2.输入regedit +回车+是 3.HKEY_CURRENT_USER\Software\Microsoft\Fiddler2 4.右键新建,选字符串值 加上 ...

  7. bzoj 3812: 主旋律 [容斥原理 状压DP]

    3812: 主旋律 题意:一张有向图,求它的生成子图是强连通图的个数.\(n \le 15\) 先说一个比较暴力的做法. 终于知道n个点图的是DAG的生成子图个数怎么求了. 暴力枚举哪些点是一个scc ...

  8. bzoj 4870: [Shoi2017]组合数问题 [矩阵乘法优化dp]

    4870: [Shoi2017]组合数问题 题意:求 \[ \sum_{i=0}^{n-1} \binom{nk}{ik+r} \mod p \] \(n \le 10^9, 0\le r < ...

  9. 2018/1/27 Zookeeper实现分布式锁

    public class DistributedClient { // 超时时间 private static final int SESSION_TIMEOUT = 5000; // zookeep ...

  10. Angular4---部署---Angular 与 Nginx的邂逅

    Nginx + Angular结合操作 1.下载Nginx , 根据自己的版本下载Nginx,关于Nginx配置,请看https://www.cnblogs.com/MBirds/p/6605366. ...