夺宝奇兵
Time Limit: 8000 ms Case Time Limit: 8000 ms Memory Limit: 64 MB
Description
HF市地下有一份宝藏,数不清的黄金、钻石……。令人遗憾的是,千年来未曾有人得到过这份宝藏。其中一个主要原因就是,要得到这份宝藏之前,首先需要通过一个迷宫。这个迷宫的结构非常复杂,一共有N个秘密房间。广为流传的一种说法是,必须在1天之内将这N个房间的门全部打开,才能找到迷宫的出口,否则就会葬身于其中。传说中同时指出,这些房间之间有着错综复杂的联系。对于每个房间而言,仅有一把能够打开这个房间的钥匙,这把钥匙可能位于这N个房间中的任意一个。每当一个房间被打开后,就可以得到房间里面的钥匙,并以打开其他的房间。被世人称为有史以来最天才的探险家小雪准备于近日进入迷宫探险。为了确保万无一失,他携带了足量的新型炸弹,每枚炸弹可以将一个房间的门直接炸掉。由于炸弹的威力太大,可能破坏整个迷宫的结果,小雪准备用最少的炸弹打开所有的房间,你能够帮助小雪计算一下吗?
Input
第一行包含一个整数N(N<=1000000),表示迷宫中房间的数目。接下来的N行,每行包含一个整数,表示能够打开第i个房间的钥匙的位置。
Output
仅包含一个整数,表示最少需要的炸弹的数目。
Sample Input
Original | Transformed |
4 2 1 2 4
Sample Output
Original | Transformed |
2
Hint
第1、3号房间的钥匙都在第2号房间里面,第2号房间的钥匙在第1号房间里面,第4号房间的钥匙就在第4号房间里面。小雪用炸弹打开第1、4号房间之后,所有的房间就都可以打开了。
Source
2006年合肥市青少年信息学(计算机)竞赛
————————————————————鸡冻的分割线————————————————————
思路:hiahiahiahiahia~~最基础的并查集,还记得上次的 连通图 吗?上次连通图是要数连接了多少次,这次正好反过来,有多少次没进行合并。嘛~时限8s?Case时限也是8s?大量输入嘛!咱们加个外挂,秒了它吧!
代码如下:
通过时间:219ms……建议修改时限变成1s……别得意了,看看出处,原来是小学生的题目……唉
————————————————————鸡冻的分割线————————————————————
思路:hiahiahiahiahia~~最基础的并查集,还记得上次的 连通图 吗?上次连通图是要数连接了多少次,这次正好反过来,有多少次没进行合并。嘛~时限8s?Case时限也是8s?大量输入嘛!咱们加个外挂,秒了它吧!
代码如下:
#include <stdio.h>
#include <string.h>
#include <math.h>
#include <stdlib.h>
int fath[1000010], bomb = 0;
void scanf_(int &num){
char in;
bool neg = false;
while(((in = getchar()) > '9'||in < '0')&&in != '-') ;
if(in == '-') {
neg = true;
while((in = getchar()) >'9'||in < '0') ;
}
num = in - '0';
while(in=getchar(), in >= '0'&&in <= '9') num *= 10, num += in - '0';
if(neg) num = 0 - num;
}
int Find(int x) {
if(x != fath[x]) fath[x] = Find(fath[x]);
return fath[x];
}
void Union(int x, int y) {
int fx = Find(x), fy = Find(y);
if(fx != fy) fath[fy] = fx;
else bomb++;
}
int main() {
int poi, a;
scanf("%d", &poi);
for(int i = 1; i <= poi; i++)
fath[i] = i;
for(int i = 1; i <= poi; i++) {
scanf_(a);
Union(i, a);
}
printf("%d\n", bomb);
return 0;
}
通过时间:219ms……建议修改时限变成1s……别得意了,看看出处,原来是小学生的题目……唉