洛谷 P2661 信息传递 Label:并查集||强联通分量
题目描述
有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 个整数,表示游戏一共可以进行多少轮。
输入输出样例
5
2 4 2 3 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:并查集||强联通分量的更多相关文章
- 洛谷 P2661 信息传递(并查集 & 最小环)
嗯... 题目链接:https://www.luogu.org/problemnew/show/P2661 这道题和一些比较水的并查集不太一样,这道题的思路就是用并查集来求最小环... 首先,如果我们 ...
- 洛谷P2661 信息传递【并查集】
题目:https://www.luogu.org/problemnew/show/P2661 题意: 有一个有向图,问最小环的的大小. 思路: 明明是图的遍历,但是bfs会T.第二组下下来的数据n居然 ...
- 洛谷P2661 信息传递 [NOIP2015] 并查集/乱搞 (待补充!
感觉我好水啊,,,做个noip往年题目还天天只想做最简单的,,,实在太菜辽 然后最水的题目还不会正解整天想着乱搞,,, 虽然也搞出来辽233333 好滴不扯辽赶紧写完去做紫题QAQ 正解:并查集 ...
- 洛谷P2661 信息传递(最小环,并查集)
洛谷P2661 信息传递 最小环求解采用并查集求最小环. 只适用于本题的情况.对于新加可以使得两个子树合并的边,总有其中一点为其中一棵子树的根. 复杂度 \(O(n)\) . #include< ...
- 洛谷P2661 信息传递——并查集
给一手链接 https://www.luogu.com.cn/problem/P2661 这道题就是 并查集求最小环 TIPS:压缩路径的时候d[x]=d[fa[x]]+d[x],而不是d[x]=d[ ...
- 洛谷 P2661 信息传递 题解
P2661 信息传递 题目描述 有 \(n\) 个同学(编号为 \(1\) 到 \(n\) )正在玩一个信息传递的游戏.在游戏里每人都有一个固定的信息传递对象,其中,编号为 \(i\) 的同学的信息传 ...
- 洛谷P2661 信息传递==coedevs4511 信息传递 NOIP2015 day1 T2
P2661 信息传递 题目描述 有n个同学(编号为1到n)正在玩一个信息传递的游戏.在游戏里每人都有一个固定的信息传递对象,其中,编号为i的同学的信息传递对象是编号为Ti同学. 游戏开始时,每人都只知 ...
- 洛谷p2661信息传递题解
题目 这个题一眼看上去就是用并查集求最小环. 我们可以设两个数组分别是f,d分别表示该点的爸爸和该点到祖先的距离. 当该点的爸爸等于他时,那他肯定就是祖先. 此时信息就肯定传递完了,此时的整个图中(我 ...
- 洛谷P2661信息传递
传送门啦 一个人要想知道自己的生日,就意味着信息的传递是成环的,因为每轮信息只能传递一个人,传递的轮数就等于环的大小 环的大小就等于环中的两个点到第三个点的距离之和加一,我们就可以在使用并查集时,维护 ...
随机推荐
- Linq学习笔记---Linq to Xml操作
LINQ to XML的成员, 属性列表: 属性 说明 Document 获取此 XObject 的 XDocument EmptySequence 获取空的元素集合 FirstAttribut ...
- GBDT原理实例演示 2
一开始我们设定F(x)也就是每个样本的预测值是0(也可以做一定的随机化) Scores = { 0, 0, 0, 0, 0, 0, 0, 0} 那么我们先计算当前情况下的梯度值 ...
- Java Eclipse进行断点调试
如何调试Java程序? 大家最开始学习Java,都会觉得IDE调试好高端有木有,其实很简单了. 下文会尽量简单直观的教会你在Eclipse中调试,其他的IDE调试步骤也是类似的. 1.在你觉得有错的地 ...
- Delphi强制类型转化和类型约定
强制类型转换时一种技术,通过它能够使编译器把一种类型的变量当做另一种类型. 由于Pascal有定义新类型的功能,因此编译器在调用一个函数时候对形参和实参类型匹配的检查是非常严格的.因此为了能够通过编译 ...
- oracle JOB学习(一)---基础
oracle job简介 下面文章来自网友(格式稍加整理) 主要的使用情景 定时在后台执行相关操作:如每天晚上0点将一张表的数据保存到另一张表中,2:定时备份数据库等 熟化说万事开头难,这 ...
- [Skills] 在桌面打开一个BAT文件,CMD窗口不关闭
每次开机都要取得本机IP,然后远程连接上去,屏幕太小,不好输入,想写个bat,执行就能看到IP,并且停留在cmd窗口上,想来简单,以前搜了好久没找到好的办法,今天找到一个贴子,竟然可以,呵呵! 以 ...
- 优秀的API接口设计原则及方法(转)
一旦API发生变化,就可能对相关的调用者带来巨大的代价,用户需要排查所有调用的代码,需要调整所有与之相关的部分,这些工作对他们来说都是额外的.如果辛辛苦苦完成这些以后,还发现了相关的bug,那对用户的 ...
- HDU 5213 Lucky 莫队+容斥
Lucky Problem Description WLD is always very lucky.His secret is a lucky number K.k is a fixed odd n ...
- loj 1002(spfa变形)
题目链接:http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=25828 题意:求所有点到给定的目标顶点的路径上的权值的最大值的最小 ...
- 廖雪峰js教程笔记4 sort排序的一些坑
排序算法 排序也是在程序中经常用到的算法.无论使用冒泡排序还是快速排序,排序的核心是比较两个元素的大小.如果是数字,我们可以直接比较,但如果是字符串或者两个对象呢?直接比较数学上的大小是没有意义的,因 ...