[Luogu 3958] NOIP2017 D2T1 奶酪
题目链接
人生第一篇题解,多多关照吧。
##注意事项:
1.多组数据,每次要*先初始化*。
2.因为涉及到开根,所以记得*开double*。
##整体思路:
建图,判断「起点」与「终点」是否连通。
方法可选择搜索(我写的BFS)或并查集(UFS)。
首先,读入时记录这些球的最小高度和最大高度,如果最低的球与底面相离,或是最高的球与顶面相离,直接Pass。
我们会发现,可能不止一个球与底面相切或相交,也可能不止一个球与顶面相切或相交。
这就是说,起点和终点都可能不止一个,这给我们操作造成了一些麻烦(然而考场上我就这么硬搜的居然AC了)。
其实,通过建立**「超级起点」**和**「超级终点」**,可以把操作变得简单——用结构体数组的第0个元素表示「超级起点」,第n+1个元素表示「超级终点」。
##两种方法都需要进行的预处理操作:
对于每一组球(i,j),计算两球球心距离是否小于半径×2。
走一遍所有的球,如果当前球可以做起点,就连上当前球和超级起点;终点亦然。
###方法1——BFS:
用二维bool数组e[i][j]记录i能否到达j,相当于存图(链式前向星也可以的,只是本题数据范围没有必要)。
对于每一个球,依次判断其能否到达0..n+1,当「超级终点」已被访问或队列已为空时结束搜索。
如果「超级终点」被访问过说明搜到了,可以到达;否则无法到达。
```cpp
#include
#include
#include
#include
#include
using namespace std;
const int MAXN=1010;
bool vis[MAXN],e[MAXN][MAXN];
double h,r,low,high;
int T,n;
struct ball
{
double x,y,z;
}s[MAXN];
double dis(ball a,ball b)
{
double t1=a.x-b.x,t2=a.y-b.y,t3=a.z-b.z;
return sqrt(t1*t1+t2*t2+t3*t3);
}
void Init()//一定记得初始化
{
low=h,high=0;
memset(vis,0,sizeof vis);
memset(e,0,sizeof e);
}
void Pre()//预处理
{
for(int i=1;i=h)//超级终点
e[n+1][i]=e[i][n+1]=1;
}
for(int i=1;i q;
q.push(0);
vis[0]=1;
while(!vis[n+1] && !q.empty())//超级终点被搜到了,或队列已空
{
int x=q.front();
q.pop();
for(int i=0;i0 || high+r
###方法2——UFS:
预处理时,如果两个球可以相连,就合并这两个球所在的UFS。
最终判断0和n+1两个球是否属于同一UFS,是则Yes,否则No。
并查集写法的代码更新于 2018.05.27,在一次水模拟赛中,原题写炸,遂重写,以前的代码风格是什么玩意!
#include <algorithm>
#include <cmath>
#include <cstdio>
using std::max;
using std::min;
const int MAXN=1010;
double h,r;
int T,n;
struct Ball
{
double x,y,z;
void Read(void)
{
scanf("%lf %lf %lf",&x,&y,&z);
}
friend double Dist(const Ball &a,const Ball &b)
{
double x=a.x-b.x,y=a.y-b.y,z=a.z-b.z;
return sqrt(x*x+y*y+z*z);
}
bool operator <(const Ball &rhs) const
{
return z<rhs.z;
}
}s[MAXN];
class UFS
{
private:
int f[MAXN];
int Find(int x)
{
return x==f[x] ? x : f[x]=Find(f[x]);
}
void Merge(int x,int y)
{
f[Find(y)]=f[Find(x)];
}
public:
UFS(int n)
{
for(int i=0;i<=n+1;++i)
f[i]=i;
for(int i=1;i<=n;++i)
{
if(s[i].z-r<=0)
Merge(0,i);
if(s[i].z+r>=h)
Merge(i,n+1);
}
for(int i=1;i<n;++i)
for(int j=i+1;j<=n;++j)
if(Dist(s[i],s[j])<=2*r)
Merge(i,j);
}
bool Connected(int x,int y)
{
return Find(x)==Find(y);
}
};
int main(int argc,char** argv)
{
scanf("%d",&T);
while(T--)
{
scanf("%d %lf %lf",&n,&h,&r);
double low=h,high=0;
for(int i=1;i<=n;++i)
{
s[i].Read();
low=min(low,s[i].z);
high=max(high,s[i].z);
}
if(low-r>0 || high+r<h)
{
puts("No");
continue;
}
UFS S(n);
puts(S.Connected(0,n+1) ? "Yes" : "No");
}
return 0;
}
/*想象一下我打完这篇文章后把文中所有的「点」一个个改成「球」*/
NOIP2017唯一AC的一道题啊。
[Luogu 3958] NOIP2017 D2T1 奶酪的更多相关文章
- NOIP2017 D2T1奶酪
这题终于是正经第一题感觉了. 只需要对相交或相切的球建一条边,然后对所有与底面有交点的球连边,再对所有与顶面有交点的球连边,bfs判断上下连通性即可. #include<iostream> ...
- NOIP2017 D2T1 奶酪
洛谷P3958 超级水的并没有用什么几何知识的几何题…… 直接爆搜一遍最后判断有没有与上/下表面相连的球之间连通即可……O(n2)不动脑子的复杂度 最多只是用一下并查集来判断两个点是否连通…… 具体细 ...
- luogu 3958 奶酪
noip2017 D2T1 奶酪 某zz选手没有想到可以用并查集来做,直接用了dijskstra,结果被ccf老爷机卡成了70分 题目大意: 现有一块大奶酪,它的高度为 h,它的长度和宽度我们可以认为 ...
- [luogu P3953] [noip2017 d1t3] 逛公园
[luogu P3953] [noip2017 d1t3] 逛公园 题目描述 策策同学特别喜欢逛公园.公园可以看成一张$N$个点$M$条边构成的有向图,且没有 自环和重边.其中1号点是公园的入口,$N ...
- [luogu P3960] [noip2017 d2t3] 队列
[luogu P3960] [noip2017 d2t3] 队列 题目描述 Sylvia 是一个热爱学习的女♂孩子. 前段时间,Sylvia 参加了学校的军训.众所周知,军训的时候需要站方阵. Syl ...
- [Luogu 3952] NOIP2017 时间复杂度
[Luogu 3952] NOIP2017 时间复杂度 一年的时间说长不长,说短,也不短. 一年之内无数次觉得难得可怕的题目,原来也就模拟这么回事儿. #include <cstdio> ...
- 【luogu】 P1433 吃奶酪
题目描述 房间里放着n块奶酪.一只小老鼠要把它们都吃掉,问至少要跑多少距离?老鼠一开始在(0,0)点处. 输入格式: 第一行一个数n (n<=15) 接下来每行2个实数,表示第i块奶酪的坐标. ...
- Luogu 3959 [NOIP2017] 宝藏
NOIP2017最后一道题 挺难想的状压dp. 受到深度的条件限制,所以一般的状态设计带有后效性,这时候考虑把深度作为一维,这样子可以保证所有状态不重复计算一遍. 神仙预处理:先处理出一个点连到一个集 ...
- Luogu 3960 [NOIP2017] 列队 - splay|线段树
题解 是我从来没有做过的裂点splay... 看的时候还是很懵逼的QAQ. 把最后一列的$n$个数放在一个平衡树中, 有 $n$ 个点 剩下的$n$行数, 每行都开一个平衡树,开始时每棵树中仅有$1$ ...
随机推荐
- DAY6敏捷冲刺
站立式会议 工作安排 (1)服务器配置 服务器端项目结构调整 (2)数据库配置 单词学习记录+用户信息 (3)客户端 客户端项目结构调整,代码功能分离 燃尽图 燃尽图有误,已重新修改,先贴卡片的界面, ...
- redis切换数据库的方法【jedis】
package com.test; import redis.clients.jedis.Jedis; public class readredis { public static void main ...
- 记一次dll强命名冲突事件
一 问题的出现 现在要做一个net分布式平台,平台涉及多个服务之间调用问题,最基础的莫过于sso.由于我们的sso采用了wcf一套私有框架实现,另外一个webapi服务通过接口调用sso服务.由于s ...
- Java问题排查工具单
前言 平时的工作中经常碰到很多疑难问题的处理,在解决问题的同时,有一些工具起到了相当大的作用,在此书写下来,一是作为笔记,可以让自己后续忘记了可快速翻阅,二是分享,希望看到此文的同学们可以拿出自己日常 ...
- EL中定义函数
1.在java类中要定义一个static函数 2配置:在WEB-INF/*.tld的配置文件 3在JSP页面上 4使用
- iOS-开发中的时间处理
做App避免不了要和时间打交道,关于时间的处理,里面有不少门道,远不是一行API调用,获取当前系统时间这么简单.我们需要了解与时间相关的各种API之间的差别,再因场景而异去设计相应的机制. 时间的形式 ...
- BZOJ 1818 内部白点(离散化+树状数组)
此题就是1227 的弱化版. 画个图或者稍微证明一下就能够知道,一定不会超过一次变换. 那么我们只需要统计有多少个白点会变黑,换句话说就是有多少个白点上下左右都有黑点. 离散化横坐标,因为没有黑点在的 ...
- BZOJ4000 TJOI2015棋盘(状压dp+矩阵快速幂)
显然每一行棋子的某种放法是否合法只与上一行有关,状压起来即可.然后n稍微有点大,矩阵快速幂即可. #include<iostream> #include<cstdio> #in ...
- [洛谷P3975][TJOI2015]弦论
题目大意:求一个字符串的第$k$大字串,$t$表示长得一样位置不同的字串是否算多个 题解:$SAM$,先求出每个位置可以到达多少个字串($Right$数组),然后在转移图上$DP$,若$t=1$,初始 ...
- POJ3498:March of the Penguins——题解
最近的题解的故事背景割. 题目: 描述 在靠近南极的某处,一些企鹅站在许多漂浮的冰块上.由于企鹅是群居动物,所以它们想要聚集到一起,在同一个冰块上.企鹅们不想把自己的身体弄湿,所以它们在冰块之间跳跃, ...