题意:有N个人互相打了M次电话,请找出所有电话圈(Eg.a→b,b→c,c→d,d→a 就算一个电话圈)并输出。(N≤25,L≤25,注意输出格式)

解法:由于N比较小所有n^2或n^3的复杂度都没有问题。所以就O(n^2)读入;O(n^3)Floyd算法求出传递闭包,d[i][j]表示 i 是否直接或间接给 j 打过电话,并查集并起一个电话圈里的人;O(n^2)输出。总的是O(n^3)的时间复杂度。

P.S.我的代码有点长~再补个连通分量和强连通分量的知识:连通分量——强连通图的连通分量为其本身。如果为非连通图,则连通分量为该图的最大连通子图;有向图强连通分量——在有向图G中,如果两个顶点vi,vj间(vi>vj)有一条从vi到vj的有向路径,同时还有一条从vj到vi的有向路径,则称两个顶点强连通(strongly connected)。如果有向图G的每两个顶点都强连通,称G是一个强连通图。有向图的极大强连通子图,称为强连通分量(strongly connected components)。

  1. 1 #include<cstdio>
  2. 2 #include<cstdlib>
  3. 3 #include<cstring>
  4. 4 #include<iostream>
  5. 5 using namespace std;
  6. 6
  7. 7 const int N=27,M=910;
  8. 8 int n,m,T=0;
  9. 9 int d[N][N],fa[N];
  10. 10 char s[N],ss[N],str[N][N];
  11. 11
  12. 12 void input()
  13. 13 {
  14. 14 int x,y,len=0;
  15. 15 memset(d,0,sizeof(d));
  16. 16 for (int i=1;i<=m;i++)
  17. 17 {
  18. 18 scanf("%s%s",s,ss);
  19. 19 x=y=0;
  20. 20 for (int j=1;j<=len;j++)
  21. 21 {
  22. 22 if (!strcmp(s,str[j])) x=j;
  23. 23 if (!strcmp(ss,str[j])) y=j;
  24. 24 }
  25. 25 if (!x) memcpy(str[++len],s,sizeof(s)),x=len;
  26. 26 if (!y) memcpy(str[++len],ss,sizeof(ss)),y=len;
  27. 27 d[x][y]=1;
  28. 28 }
  29. 29 }
  30. 30 int ffind(int x)
  31. 31 {
  32. 32 if (fa[x]!=x) fa[x]=ffind(fa[x]);
  33. 33 return fa[x];
  34. 34 }
  35. 35 void solve()
  36. 36 {
  37. 37 for (int k=1;k<=n;k++)
  38. 38 for (int i=1;i<=n;i++)
  39. 39 for (int j=1;j<=n;j++)
  40. 40 d[i][j]=d[i][j]|(d[i][k]&d[k][j]);
  41. 41
  42. 42 for (int i=1;i<=n;i++) fa[i]=i;
  43. 43 for (int i=1;i<=n;i++)
  44. 44 for (int j=1;j<=n;j++)
  45. 45 {
  46. 46 if (!d[i][j] || !d[j][i]) continue;
  47. 47 int fx=ffind(i),fy=ffind(j);
  48. 48 if (fx!=fy) fa[fx]=fy;
  49. 49 }
  50. 50 if (T) printf("\n");
  51. 51 printf("Calling circles for data set %d:\n",++T);
  52. 52 for (int i=1;i<=n;i++)
  53. 53 {
  54. 54 if (fa[i]!=i) continue;
  55. 55 printf("%s",str[i]);
  56. 56 for (int j=1;j<=n;j++)
  57. 57 if (fa[j]==i && i!=j) printf(", %s",str[j]);
  58. 58 printf("\n");
  59. 59 }
  60. 60 }
  61. 61 int main()
  62. 62 {
  63. 63 while (1)
  64. 64 {
  65. 65 scanf("%d%d",&n,&m);
  66. 66 if (!n && !m) break;
  67. 67 input();
  68. 68 solve();
  69. 69 }
  70. 70 return 0;
  71. 71 }

【uva 247】Calling Circles(图论--Floyd 传递闭包+并查集 连通分量)的更多相关文章

  1. UVa 247 - Calling Circles(Floyd求有向图的传递闭包)

    链接: https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem& ...

  2. UVA - 247 Calling Circles(Floyd求传递闭包)

    题目: 思路: 利用Floyd求传递闭包(mp[i][j] = mp[i][j]||(mp[i][k]&&mp[k][j]);),当mp[i][j]=1&&mp[j][ ...

  3. UVA 247"Calling Circles"(floyd求传递闭包+SCC)

    传送门 题意: 如果两个人相互打电话(直接或间接),则说他们在同一个电话圈里. (a,b) 表示 a 打给 b: 例如,(a,b),(b,c),(c,d),(d,a),则这四个人在同一个电话圈里: 输 ...

  4. UVA 247 - Calling Circles (Floyd)

    互相可以打电话是一个传递关系,所以Floyd求传递封包,dfs找一个尽量大的圈. #include<bits/stdc++.h> using namespace std; ; map< ...

  5. UVa 247 Calling Circles【传递闭包】

    题意:给出n个人的m次电话,问最后构成多少个环,找出所有的环 自己想的是:用map来储存人名,每个人名映射成一个数字编号,再用并查集,求出有多少块连通块,输出 可是map不熟,写不出来,而且用并查集输 ...

  6. UVA - 247 Calling Circles Floyd判圈

    思路:利用的Floyd判圈,如果i能到j,j也能到i说明i和j在同一个圈里.每个人的名字可用map编号.最后DFS打印答案即可. AC代码 #include <cstdio> #inclu ...

  7. UVa 247 Calling Circles (DFS+Floyd)

    题意:如果两个人互通电话,那么他们就在一个电话圈里,现在给定 n 个人,并且给定 m 个通话记录,让你输出所有的电话圈. 析:刚开始没想到是Floyd算法,后来才知道是这个算法,利用这个算法进行连通性 ...

  8. UVA 247 电话圈 (floyd传递闭包 + dfs输出连通分量的点)

    题意:输出所有的环: 思路:数据比较小,用三层循环的floyd传递闭包(即两条路通为1,不通为0,如果在一个环中,环中的所有点能互相连通),输出路径用dfs,递归还没有出现过的点(vis),输出并递归 ...

  9. UVa 247 电话圈(Floyd传递闭包)

    https://vjudge.net/problem/UVA-247 题意: 如果两个人相互打电话,则说他们在同一个电话圈里.例如,a打给b,b打给c,c打给d,d打给a,则这4个人在同一个圈里:如果 ...

随机推荐

  1. 记一次Goroutine与wg导致的问题

    前言 今天发现了一个问题是之前一直没有注意到的,这里记一下 正文 Send Closed Chan 问题概述 代码逻辑是启动时启动多个 channel, channel1 获取数据监听数据处理后发送给 ...

  2. Jenkins-自动部署,备份

    Jenkins-自动部署,备份 问题导入: 环境: CentOS 7,   Tomcat 8.5,   Jdk 1.8,   Maven 3.25 ,Jenkins war包 2.x 原因: 每次部署 ...

  3. LeetCode235 二叉搜索树的最近公共祖先

    给定一个二叉搜索树, 找到该树中两个指定节点的最近公共祖先. 百度百科中最近公共祖先的定义为:"对于有根树 T 的两个结点 p.q,最近公共祖先表示为一个结点 x,满足 x 是 p.q 的祖 ...

  4. 【Linux】Linux下如何分区及如何格式化

    环境:CentOS7.1 磁盘大小是1.8T 将磁盘/dev/sda分一个分区,分区类型为xfs fdisk /dev/sda n    --创建新分区 p  --创建分区类型为主分区 1  --主分 ...

  5. postgresql数据库升级

    pg_upgrade官网介绍:https://www.postgresql.org/docs/10/pgupgrade.html 1.查看老版本数据库编译参数值并记录 select name,sett ...

  6. DSL是什么?Elasticsearch的Query DSL又是什么?

    1.DSL简介 DSL 其实是 Domain Specific Language 的缩写,中文翻译为领域特定语言.而与 DSL 相对的就是 GPL,这里的 GPL 并不是我们知道的开源许可证(备注:G ...

  7. java虚拟机入门(一)-jvm基础

    转行学java之前,总是听着大佬们说着java像个渣男一样可以跨平台,一次编译到处运行,瞬间,我就坚定了学java的信念,哎呀妈呀,得劲.真的学java之后,好像渣男也不是那么好学的,尤其这货的必杀技 ...

  8. 前端面试之JavaScript中数组的方法!【残缺版!!】

    前端面试之JavaScript中数组常用的方法 7 join Array.join()方法将数组中所有元素都转化为字符串并连接在-起,返回最后生成的字 符串.可以指定一个可选的字符串在生成的字符串中来 ...

  9. Linux系统中的Page cache和Buffer cache

    Linux系统中的Page cache和Buffer cache Linux中有两个很容易混淆的概念,pagecache和buffercache,首先简单将一些Linux系统下内存的分布,使用free ...

  10. Git:.gitignore和.gitkeep文件的使用 让空文件夹被跟踪

    Git:.gitignore和.gitkeep文件的使用 Git:.gitignore和.gitkeep文件的使用 https://majing.io/posts/10000001781172 .gi ...