题目描述

有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。

代码

 #include<iostream>
#include<cstring>
#include<cstdio>
#include<algorithm>
#include<vector>
#define INF 0x3f3f3f3f
using namespace std;
vector<int> G[],rG[],vs;
int N,tot=INF,ans,used[],cmp[],dep[],rank[],pa[];int k=;
//并查集
void init_bingchaji(){
for(int i=;i<=;i++){
rank[i]=;pa[i]=i;
}
}
int Find(int a){
if(pa[a]==a) return a;
else return pa[a]=Find(pa[a]);
}
void unite(int a,int b){
int ta=Find(a),pb=Find(b);
if(rank[ta]>rank[pb]) pa[pb]=ta;
else{
pa[ta]=b;
if(rank[ta]==rank[pb]) rank[pb]++;
}
} //强联通分量
void init_qingliantong(){
;
}
void add_edge(int a,int b){
G[a].push_back(b);
rG[b].push_back(a);
}
void dfs(int v){
used[v]=;
for(int i=;i<G[v].size();i++){
if(!used[G[v][i]]) dfs(G[v][i]);
}
vs.push_back(v);
}
void rdfs(int v,int k,int depth){
used[v]=;
dep[v]=depth;
cmp[v]=k;
for(int i=;i<rG[v].size();i++){
if(!used[rG[v][i]]) rdfs(rG[v][i],k,depth+); int& now=rG[v][i];
if(cmp[v]==cmp[now]&&Find(v)==Find(now)){
for(int j=;j<G[v].size();j++){
// printf("v=%d G[v][j]=%d \n",v,G[v][j]);
if(Find(now)==Find(G[v][j])){
tot=min(tot,dep[G[v][j]]-dep[v]+);
// printf("%d %d\n",v,G[v][j]);
}
}
}
else unite(v,rG[v][i]);
}
for(int i=;i<rG[v].size();i++){
if(cmp[v]==cmp[rG[v][i]]) continue; }
}
void scc(){
for(int i=;i<=N;i++){
if(!used[i]) dfs(i);
}
// for(int i=0;i<vs.size();i++) printf("%d ",vs[i]);
memset(used,,sizeof(used));
init_bingchaji();
for(int i=vs.size()-;i>=;i--){
if(!used[vs[i]]) {
// init_bingchaji();
pa[vs[i]]=vs[i];
rdfs(vs[i],k++,);
}
}
} int main(){
// freopen("01.in","r",stdin);
scanf("%d",&N);
for(int i=;i<=N;i++){
int tmp;scanf("%d",&tmp);
add_edge(i,tmp);
}
scc();
// for(int i=1;i<=5;i++) printf("%d\n",cmp[i]);
printf("%d\n",tot);
}

非常恶心的 强联通

以下为并查集特别版

 #include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<queue> using namespace std; queue <int>q;
int n,ans,father[],vis[],ent[]; int main()
{
cin>>n;
ans=n;
int x,i,j;
for(int i=;i<=n;++i)
{
scanf("%d",&father[i]);
++ent[father[i]];
}
for(int i=;i<=n;++i)
{
if(ent[i]==)
{
q.push(i);
vis[i]=;
}
}
while(!q.empty())
{
x=q.front();
q.pop();
--ent[father[x]];
if(ent[father[x]]==)
{
vis[father[x]]=;
q.push(father[x]);
}
}
for(int i=;i<=n;++i)
{
if(ent[i]!=&&vis[i]!=)
{
vis[i]=;
j=father[i];
x=;
while(vis[j]==)
{
vis[j]=;
j=father[j];
++x;
}
if(ans>=x)ans=x;
} }
printf("%d",ans);
puts("");
return ;
}

洛谷 P2661 信息传递 Label:并查集||强联通分量的更多相关文章

  1. 洛谷 P2661 信息传递(并查集 & 最小环)

    嗯... 题目链接:https://www.luogu.org/problemnew/show/P2661 这道题和一些比较水的并查集不太一样,这道题的思路就是用并查集来求最小环... 首先,如果我们 ...

  2. 洛谷P2661 信息传递【并查集】

    题目:https://www.luogu.org/problemnew/show/P2661 题意: 有一个有向图,问最小环的的大小. 思路: 明明是图的遍历,但是bfs会T.第二组下下来的数据n居然 ...

  3. 洛谷P2661 信息传递 [NOIP2015] 并查集/乱搞 (待补充!

    感觉我好水啊,,,做个noip往年题目还天天只想做最简单的,,,实在太菜辽 然后最水的题目还不会正解整天想着乱搞,,,  虽然也搞出来辽233333 好滴不扯辽赶紧写完去做紫题QAQ 正解:并查集  ...

  4. 洛谷P2661 信息传递(最小环,并查集)

    洛谷P2661 信息传递 最小环求解采用并查集求最小环. 只适用于本题的情况.对于新加可以使得两个子树合并的边,总有其中一点为其中一棵子树的根. 复杂度 \(O(n)\) . #include< ...

  5. 洛谷P2661 信息传递——并查集

    给一手链接 https://www.luogu.com.cn/problem/P2661 这道题就是 并查集求最小环 TIPS:压缩路径的时候d[x]=d[fa[x]]+d[x],而不是d[x]=d[ ...

  6. 洛谷 P2661 信息传递 题解

    P2661 信息传递 题目描述 有 \(n\) 个同学(编号为 \(1\) 到 \(n\) )正在玩一个信息传递的游戏.在游戏里每人都有一个固定的信息传递对象,其中,编号为 \(i\) 的同学的信息传 ...

  7. 洛谷P2661 信息传递==coedevs4511 信息传递 NOIP2015 day1 T2

    P2661 信息传递 题目描述 有n个同学(编号为1到n)正在玩一个信息传递的游戏.在游戏里每人都有一个固定的信息传递对象,其中,编号为i的同学的信息传递对象是编号为Ti同学. 游戏开始时,每人都只知 ...

  8. 洛谷p2661信息传递题解

    题目 这个题一眼看上去就是用并查集求最小环. 我们可以设两个数组分别是f,d分别表示该点的爸爸和该点到祖先的距离. 当该点的爸爸等于他时,那他肯定就是祖先. 此时信息就肯定传递完了,此时的整个图中(我 ...

  9. 洛谷P2661信息传递

    传送门啦 一个人要想知道自己的生日,就意味着信息的传递是成环的,因为每轮信息只能传递一个人,传递的轮数就等于环的大小 环的大小就等于环中的两个点到第三个点的距离之和加一,我们就可以在使用并查集时,维护 ...

随机推荐

  1. CLR via C#(11)-无参属性、有参数属性(索引器)

    一. 无参属性 1. 定义属性 无参属性就是我们最常见的属性方式,在赋值时可以加入一定的逻辑判断.属性的定义其实不复杂,先看个直观的例子: 说明: 属性要定义名称和类型,且类型不能是void. 属性是 ...

  2. 数据结构和算法 – 8.链表

    8.1.数组存在的问题 在处理列表的时候数组是常用的数据结构.数组可以对所存储的数据项提供快速地存取访问,而且它很易于进行循环遍历操作.当然,数组已经是语言的一部分了,用户不需要使用额外的内存,也不需 ...

  3. Hibernate的ORM原理和实现

    >>Hibernate和ORM ORM的全称是Object Relational Mapping,即对象关系映射.它的实现思想就是将关系数据库中表的数据映射成为对象,以对象的形式展现,这样 ...

  4. python中最简单的多进程程序

    学着.. #!/usr/bin/env python # -*- coding: utf-8 -*- # Spawn a Process: Chapter 3: Process Based Paral ...

  5. Java简明教程

    Java与C++比较概况 C++ Java class Foo { // 声明 Foo 类 public: int x; // 成员变量 Foo(): x() { // Foo 的构造函数Constr ...

  6. php重修

    阅读顺序: http://www.laruence.com/2008/08/11/147.html  深入浅出php http://www.laruence.com/2008/06/18/221.ht ...

  7. Unreal Engine4 学习笔记2 动画蒙太奇

    动画蒙太奇出现的位置是在动画蓝图的动画图表和事件图表中,如下图 事件图表,可以看出在主线执行的结尾,如果is Punching 为true,则会执行一个我们自定义的Punch Event,用来播放动画 ...

  8. Mysql常用命令详解

    Mysql安装目录 数据库目录 /var/lib/mysql/ 配置文件 /usr/share/mysql(mysql.server命令及配置文件) 相关命令 /usr/bin(mysqladmin ...

  9. Objective-C编码规范:26个方面解决iOS开发问题(转)

    链接

  10. Java优化之输出十万以内的质数

    (1)未经优化时所耗费的时间: public class PrimeNumber { public static void main(String[] args) { long start = Sys ...