【 NOIP2015 DAY1 T2 信息传递】带权并查集

题目描述

有n个同学(编号为1到n)正在玩一个信息传递的游戏。在游戏里每人都有一个固定的信息传递对象,其中,编号为i的同学的信息传递对象是编号为Ti同学。

游戏开始时,每人都只知道自己的生日。之后每一轮中,所有人会同时将自己当前所知的生日信息告诉各自的信息传递对象(注意:可能有人可以从若干人那里获取信息,但是每人只会把信息告诉一个人,即自己的信息传递对象)。当有人从别人口中得知自己的生日时,游戏结束。请问该游戏一共可以进行几轮?

输入输出格式

输入格式:

输入共2行。

第1行包含1个正整数n表示n个人。

第2行包含n个用空格隔开的正整数T1,T2,……,Tn其中第i个整数Ti示编号为i

的同学的信息传递对象是编号为Ti的同学,Ti≤n且Ti≠i

数据保证游戏一定会结束。

输出格式:

输出共 1 行,包含 1 个整数,表示游戏一共可以进行多少轮。

输入输出样例

输入样例#1:

5
2 4 2 3 1

输出样例#1:

3

说明

样例1解释

游戏的流程如图所示。当进行完第 3 轮游戏后, 4 号玩家会听到 2 号玩家告诉他自

己的生日,所以答案为 3。当然,第 3 轮游戏后, 2 号玩家、 3 号玩家都能从自己的消息

来源得知自己的生日,同样符合游戏结束的条件。

对于 30%的数据, n ≤ 200;

对于 60%的数据, n ≤ 2500;

对于 100%的数据, n ≤ 200000。

表示我上一年智障,今年也智障、、

其实这是一个特殊的图,n个点 n条边 并且每个点出度都为1

也就是说,每个联通块只有一个环,并且如果你无视方向的话找到的环,考虑方向之后还是一个环(因为不会有一个点有两条出边)

所以其实可以直接并查集过,我带了一个权,记录两点的距离。

 1 #include<cstdio>
 2 #include<cstdlib>
 3 #include<cstring>
 4 #include<iostream>
 5 #include<algorithm>
 6 #include<queue>
 7 #include<cmath>
 8 using namespace std;
 9 #define Maxn 200010
10 #define INF 0xfffffff
11
12 int d[Maxn],fa[Maxn];
13
14 int ffa(int x)
15 {
16     int now=fa[x];
17     if(x!=fa[x]) fa[x]=ffa(fa[x]),d[x]+=d[now];
18     return fa[x];
19 }
20
21 int mymin(int x,int y) {return x<y?x:y;}
22
23 int main()
24 {
25     freopen("a.in","r",stdin);
26     freopen("a.out","w",stdout);
27     int n;
28     scanf("%d",&n);
29     for(int i=1;i<=n;i++) fa[i]=i,d[i]=0;
30     int ans=INF;
31     for(int i=1;i<=n;i++)
32     {
33         int x;
34         scanf("%d",&x);
35         if(ffa(x)==ffa(i))
36         {
37             ans=mymin(ans,d[x]+d[i]+1);
38         }
39         else
40         {
41             int now=ffa(i);
42             fa[now]=i;d[now]=d[i];
43             d[i]=1;fa[i]=x;
44         }
45     }
46     printf("%d\n",ans);
47     return 0;
48 }

2016-11-15 11:24:19

时间: 2024-11-05 06:27:41

【 NOIP2015 DAY1 T2 信息传递】带权并查集的相关文章

poj1984 带权并查集

题意:有多个点,在平面上位于坐标点上,给出一些关系,表示某个点在某个点的正东/西/南/北方向多少距离,然后给出一系列询问,表示在第几个关系给出后询问某两点的曼哈顿距离,或者未知则输出-1. 只要在元素的权值上保存两个信息,与祖先元素的两个方向的差,我选择正东和正北方向差(负值表示正西和正南),然后直接用带权并查集,询问时曼哈顿距离就是两个权值的绝对值之和.由于询问是嵌在给出关系中间的,所以要先存下所有关系和询问,离线做就行. 1 #include<stdio.h> 2 #include<

带权并查集

似乎好些天没写文章了,感慨万千啊. 一如既往,先介绍一下题目的需求吧,如下: 题目简述:某人想编写一个字典,该字典中存在正反两种关系,并且这种关系具有某种规则下的传递性质,如下: A 与 B 同义, B 与 C 同义,则 A 与 C同义 : A 与 B 同义, B 与 C 反义,则 A 与 C反义 : A 与 B 反义, B 与 C 同义,则 A 与 C反义 : A 与 B 反义, B 与 C 反义,则 A 与 C同义 . 此人最初定义部分字符串的正反关系,要求以此为基础将所有给定字符串之间的关

[NOIP摸你赛]Hzwer的陨石(带权并查集)

题目描述: 经过不懈的努力,Hzwer召唤了很多陨石.已知Hzwer的地图上共有n个区域,且一开始的时候第i个陨石掉在了第i个区域.有电力喷射背包的ndsf很自豪,他认为搬陨石很容易,所以他将一些区域的陨石全搬到了另外一些区域. 在ndsf愉快的搬运过程中,Hzwer想知道一些陨石的信息.对于Hzwer询问的每个陨石i,你必须告诉他,在当前这个时候,i号陨石在所在区域x.x区域共有的陨石数y.以及i号陨石被搬运的次数z. 输入描述: 输入的第一行是一个正整数T.表示有多少组输入数据. 接下来共有

并查集2——带权并查集

路径压缩 前面的并查集的复杂度实际上有些极端情况会很慢.比如树的结构正好是一条链,那么最坏情况下,每次查询的复杂度达到了 O(n). 路径压缩 的思想是,我们只关心每个结点的父结点,而并不太关心树的真正的结构. 这样我们在一次查询的时候,可以把查询路径上的所有结点的 father[i] 都赋值成为根结点.只需要在我们之前的查询函数上面很小的改动. int get(int x) { if (father[x] == x) { // x 结点就是根结点 return x; } return fath

hdu3038 How Many Answers Are Wrong【带权并查集】

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3038 题目描述:某个无聊的骚年给他的女友(烧!)一系列三元组x,y,c,表示序列中ax+ax+1+...+ay=c.但是其中有一些是错误的,也就是与前面的信息出现了冲突,比如给了1,2,6和3,4,5接着却给了1,4,100.找出所有错误的三元组的数量 思路:正如一般带权并查集的方法.用par[i]记录父节点,d[i]记录与父节点的差值,如果x与y在不同的集合中则可以自由合并,如果在同一集合中那就有

【POJ 1984】Navigation Nightmare(带权并查集)

Navigation Nightmare Description Farmer John's pastoral neighborhood has N farms (2 <= N <= 40,000), usually numbered/labeled 1..N. A series of M (1 <= M < 40,000) vertical and horizontal roads each of varying lengths (1 <= length <= 100

poj1984 带权并查集(向量处理)

Navigation Nightmare Time Limit: 2000MS   Memory Limit: 30000K Total Submissions: 5939   Accepted: 2102 Case Time Limit: 1000MS Description Farmer John's pastoral neighborhood has N farms (2 <= N <= 40,000), usually numbered/labeled 1..N. A series o

洛谷OJ P1196 银河英雄传说(带权并查集)

题目描述 公元五八○一年,地球居民迁移至金牛座α第二行星,在那里发表银河联邦 创立宣言,同年改元为宇宙历元年,并开始向银河系深处拓展. 宇宙历七九九年,银河系的两大军事集团在巴米利恩星域爆发战争.泰山压 顶集团派宇宙舰队司令莱因哈特率领十万余艘战舰出征,气吞山河集团点名将杨 威利组织麾下三万艘战舰迎敌. 杨威利擅长排兵布阵,巧妙运用各种战术屡次以少胜多,难免恣生骄气.在 这次决战中,他将巴米利恩星域战场划分成30000列,每列依次编号为1, 2, …, 30000.之后,他把自己的战舰也依次编号

带权并查集 POJ1988 POJ2492

单纯的并查集很简单,带权并查集还能解决更多的问题,才更好玩,来个题热身.对于下面的知识,现在就当你已经熟练掌握了递归和并查集的路径压缩. POJ1988:题目链接 http://poj.org/problem?id=1988 题目大意:有N(N<=30,000)堆方块,开始每堆都是一个方块.方块编号1 – N. 有两种操作:  M x y : 表示把方块x所在的堆,拿起来叠放到y所在的堆上.  C x : 问方块x下面有多少个方块. 操作最多有 P (P<=100,000)次.对每次C操作