http://hihocoder.com/problemset/problem/1291

前几天比较忙,这次来补一下微软笔试的最后一题,这题是这次微软笔试的第四题,过的人比较少,我当时在调试B题,没时间看这一题。不过打过之前一场BestCoder的应该都会有点思路,虽然BC那题是二维,这题是三维的,但是思路应该是一样的,没错,就是离线加并查集。

正过来考虑的时候,发现第一个要求相邻块是好处理的,但是第二个要求能否到达(1000, 1000, 1000)这个条件似乎比较难判断,当时BC上的题根据题意是可以二分加搜索,但是这题的条件是不能二分的。

离线并查集的话,比较好想(如果做过BC那题的话),就是先算上所有块,搜索一遍全图,把非块的用并查集联通起来。然后倒着来拆块,看要拆的块是否和边界(这里取了(0, 0, 1)这个边界非块)的非块联通(有公共的根),然后将拆掉块的地方和周围非块的地方联通。直到某一处的块与边界不联通,那么就是No。否则最后就是Yes。

本地用dfs爆栈了,改bfs应该可以解决。不过直接交上去,评测机用dfs不会爆栈。。。

代码:

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <queue>
#include <map>
#include <set>
#include <string>
#include <vector> using namespace std; const int maxN = ;
const int maxX = ;
int ufs[**];
int n, x[maxN], y[maxN], z[maxN];
int xx[] = {-, , , , , };
int yy[] = { , , -, , , };
int zz[] = { , , , , -, }; int findRoot(int x)
{
if (ufs[x] == x) return x;
int fa = findRoot(ufs[x]);
ufs[x] = fa;
return fa;
} void mergeUfs(int x, int y)
{
int fx, fy;
fx = findRoot(x);
fy = findRoot(y);
ufs[fx] = fy;
} bool inSameUfs(int x, int y)
{
int fx, fy;
fx = findRoot(x);
fy = findRoot(y);
if (fx == fy) return true;
else return false;
} inline int Hash(int x, int y, int z)
{
return x*maxX*maxX+y*maxX+z;
} void dfs(int x, int y, int z)
{
int t = Hash(x, y, z), tt;
if (ufs[t] == - || ufs[t] != -) return;
if (ufs[t] == -) ufs[t] = t;
for (int i = ; i < ; ++i)
{
if (x+xx[i] < || y+yy[i] < || z+zz[i] <= ) continue;
if (x+xx[i] >= maxX || y+yy[i] >= maxX || z+zz[i] >= maxX) continue;
tt = Hash(x+xx[i], y+yy[i], z+zz[i]);
if (ufs[tt] != -) continue;
dfs(x+xx[i], y+yy[i], z+zz[i]);
mergeUfs(t, tt);
}
} bool judge(int x, int y, int z)
{
bool flag = false;
int t, tt, to;
for (int i = ; i < ; ++i)
{
tt = Hash(x+xx[i], y+yy[i], z+zz[i]);
if (ufs[tt] == -) flag = true;
}
if (!flag) return false; flag = false;
t = Hash(x, y, z);
to = Hash(, , );
ufs[t] = t;
for (int i = ; i < ; ++i)
{
tt = Hash(x+xx[i], y+yy[i], z+zz[i]);
if (ufs[tt] == -) continue;
if (findRoot(to) == findRoot(tt)) flag = true;
mergeUfs(t, tt);
}
return flag;
} void input()
{
memset(ufs, -, sizeof(ufs));
int t;
scanf("%d", &n);
for (int i = ; i < n; ++i)
{
scanf("%d%d%d", &x[i], &y[i], &z[i]);
t = Hash(x[i], y[i], z[i]);
ufs[t] = -;
}
for (int i = ; i < maxX; ++i)
for (int j = ; j < maxX; ++j)
{
t = Hash(i, j, );
ufs[t] = -;
}
for (int k = ; k < maxX; ++k)
for (int i = ; i < maxX; ++i)
for (int j = ; j < maxX; ++j)
dfs(i, j, k);
} void work()
{
for (int i = n-; i >= ; i--)
{
if (!judge(x[i], y[i], z[i]))
{
printf("No\n");
return;
}
}
printf("Yes\n");
} int main()
{
//freopen("test.in", "r", stdin);
//freopen("test.out", "w", stdout);
int T;
scanf("%d", &T);
for (int times = ; times <= T; ++times)
{
input();
work();
}
return ;
}

ACM学习历程—Hihocoder 1291 Building in Sandbox(dfs && 离线 && 并查集)的更多相关文章

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

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

  2. ACM学习历程—Hihocoder 1139 二分·二分答案(bfs)

    http://hihocoder.com/problemset/problem/1139 这题提示上写的是二分,但是感觉不二分应该也可以,至少题目是AC的... 二分的思想就是二分答案的值,看能不能在 ...

  3. ACM学习历程—Hihocoder 1289 403 Forbidden(字典树 || (离线 && 排序 && 染色))

    http://hihocoder.com/problemset/problem/1289 这题是这次微软笔试的第二题,过的人比第三题少一点,这题一眼看过去就是字符串匹配问题,应该可以使用字典树解决.不 ...

  4. ACM学习历程—Hihocoder 1290 Demo Day(动态规划)

    http://hihocoder.com/problemset/problem/1290 这题是这次微软笔试的第三题,过的人比第一题少一点,这题一眼看过去就是动态规划,不过转移方程貌似不是很简单,调试 ...

  5. ACM学习历程—Hihocoder 1288 Font Size(暴力 || 二分)

    http://hihocoder.com/problemset/problem/1288 这题是这次微软笔试的第一题,关键的是s的上限是min(w, h),这样s的范围只有到1000,这样就可以直接暴 ...

  6. ACM学习历程—Hihocoder [Offer收割]编程练习赛1

    比赛链接:http://hihocoder.com/contest/hihointerview3/problem/1 大概有一个月没怎么打算法了.这一场的前一场BC,也打的不是很好.本来Div1的A和 ...

  7. ACM学习历程—Hihocoder 1233 Boxes(bfs)(2015北京网赛)

    hihoCoder挑战赛12 时间限制:1000ms 单点时限:1000ms 内存限制:256MB   描述 There is a strange storehouse in PKU. In this ...

  8. ACM学习历程—Hihocoder 1164 随机斐波那契(数学递推)

    时间限制:5000ms 单点时限:1000ms 内存限制:256MB 描述 大家对斐波那契数列想必都很熟悉: a0 = 1, a1 = 1, ai = ai-1 + ai-2,(i > 1). ...

  9. ACM学习历程—Hihocoder 1178 计数(位运算 && set容器)(hihoCoder挑战赛12)

    时间限制:10000ms 单点时限:1000ms 内存限制:256MB   描述 Rowdark是一个邪恶的魔法师.在他阅读大巫术师Lich的传记时,他发现一类黑魔法来召唤远古生物,鱼丸. 魔法n能召 ...

随机推荐

  1. linux中安装软件的集中方法

    一.rpm包安装方式步骤: 引用:1.找到相应的软件包,比如soft.version.rpm,下载到本机某个目录:2.打开一个终端,su -成root用户:3.cd soft.version.rpm所 ...

  2. 计算机网络概述---传输层 UDP和TCP

    传输层的功能 传输层为应用进程间提供端到端的逻辑通信(网络层是提供主机之间的逻辑通信), 传输层两大重要的功能:复用 和 分用. 复用:在发送端,多个应用进程公用一个传输层: 分用:在接收端,传输层会 ...

  3. Android开发BUG及解决方法

    错误描述 问题1: 按照提示打开gradle-wrapper.properties文件 并且将gradle-2.8-all.zip改为gradle-2.10-all.zip,重新导入项目 问题2: 却 ...

  4. 【HackerRank】Find the Median(Partition找到数组中位数)

    In the Quicksort challenges, you sorted an entire array. Sometimes, you just need specific informati ...

  5. linux alsa pcm(此pcm非硬件pcm接口)

    转:https://blog.csdn.net/crycheng/article/details/7095899 CODEC :音频芯片的控制,比如静音.打开(关闭)ADC(DAC).设置ADC(DA ...

  6. Qt事件机制---信号通过事件实现,事件可以过滤,事件更底层,事件是基础,信号是扩展。

    转:http://www.cnblogs.com/findumars/p/8001484.html Qt事件机制(是动作发生后,一种通知对象的消息,是被动与主动的总和.先处理自己队列中的消息,然后再处 ...

  7. MVC6 OWin Microsoft Identity 自定义验证

    1. Startup.cs中修改默认的验证设置 //app.UseIdentity(); app.UseCookieAuthentication(options => { //options.A ...

  8. 16个tomcat面试题

    1)解释什么是Jasper? Jasper是Tomcat的JSP引擎 它解析JSP文件,将它们编译成JAVA代码作为servlet 在运行时,Jasper允许自动检测JSP文件的更改并重新编译它们 2 ...

  9. Oracle 数据库 INTERVAL DAY TO SECOND类型的使用

    INTERVAL DAY TO SECOND类型可以用来存储单位为天和秒的时间间隔.下面这条语句创建一个名为promotions的表,用来存储促销信息.promotions表包含了一个INTERVAL ...

  10. DataX的安装

    DataX的安装 1. 可下载tar包 https://github.com/alibaba/DataX/blob/master/userGuid.md 2. 下载源码自己编译 git clone h ...