[leetcode/lintcode 题解] 谷歌面试题:找出有向图中的弱连通分量
请找出有向图中弱连通分量。图中的每个节点包含 1 个标签和1 个相邻节点列表。(有向图的弱连通分量是任意两点均有有向边相连的极大子图)
将连通分量内的元素升序排列。
图模型说明:
Graph
For example:
{1,2,4#2,1,4#3,5#4,1,2#5,3} represents follow graph:
------
\ | |
\ | |
\ | |
\ | |
we use # to split each node information.
1,2,4 represents that 2, 4 are 1's neighbors
2,1,4 represents that 1, 4 are 2's neighbors
3,5 represents that 5 is 3's neighbor
4,1,2 represents that 1, 2 are 4's neighbors
5,3 represents that 3 is 5's neighbor
样例 1:
输入: {,,#,#,###,}
输出: [[,,],[,,]]
解释:
-----> -->
\ | ^
\ | |
\ |
\ v
->
样例 2:
输入: {,#,#,}
输出: [[,,]]
【题解】
算法:并查集
并查集是一种可以动态维护若干个不重叠的集合,并支持合并与查询两种操作的一种数据结构
一般我们建立一个数组fa或者用map表示一个并查集,fa[i]表示i的父节点。
- 初始化:每一个点都是一个集合,因此自己的父节点就是自己fa[i]=i
- 查询:每一个节点不断寻找自己的父节点,若此时自己的父节点就是自己,那么该点为集合的根结点,返回该点。
- 修改:合并两个集合只需要合并两个集合的根结点,即fa[RootA]=RootB,其中RootA,RootB是两个元素的根结点。
- 路径压缩:实际上,我们在查询过程中只关心根结点是什么,并不关心这棵树的形态(有一些题除外)。因此我们可以在查询操作的时候将访问过的每个点都指向树根,这样的方法叫做路径压缩,单次操作复杂度为O(logn),经过路径压缩后可将查询时间优化到O(1)
我们用um作为father数组维护点之间的关系,通过并查集连通块,将有向连接转换为无向连接
思路
对于图上上两个点,只要他们有连边关系,那么他们就属于同一个弱连通分量
于是这道题我们就可以转换为无向图找联通块
算法流程
- 遍历图上节点,对于有连边关系的两个点,通过并查集将他们合并为一个合集
- 遍历并查集中的每一个点,找到其父亲,再加入父亲所在的 List 中
- 将所有 List 按照点从小到大排序输出
复杂度分析
- 时间复杂度
O(n) - 并查集的复杂度
- 空间复杂度
O(n) - 需要用一个map去记录点与点之间的关系
public class Solution {
/*
* @param nodes: a array of Directed graph node
* @return: a connected set of a directed graph
*/
private Map<Integer, Integer> um=null;
public List<List<Integer>> connectedSet2(ArrayList<DirectedGraphNode> nodes) {
um = new HashMap<>();
for (DirectedGraphNode node : nodes) {//将所有连通点用并查集建立图
find(node.label);
for (DirectedGraphNode neighbor : node.neighbors) {
connect(node.label, neighbor.label);
}
}
Map<Integer, List<Integer>> hash = new HashMap<>();
for (Integer roots : um.keySet()) {//查找每个节点所属的块
int father = find(roots);
if (!hash.containsKey(father)) {
hash.put(father, new ArrayList<>());
}
List<Integer> list = hash.get(father);
list.add(roots);
Collections.sort(list);
}
return new ArrayList<>(hash.values());
}
private int find (int x) {//查询函数
if (!um.containsKey(x)) {
um.put(x,x);
}
int father = um.get(x);
if (father == x) {
return x;
}
father = find(father);
um.put(x, father);
return father;
}
private void connect(int a, int b) {//合并函数
int roota = find(a),rootb=find(b);
if (roota != rootb) {
um.put(Math.min(roota, rootb), Math.max(roota, rootb));
}
}
}
[leetcode/lintcode 题解] 谷歌面试题:找出有向图中的弱连通分量的更多相关文章
- [leetcode/lintcode 题解] 微软面试题:股票价格跨度
编写一个 StockSpanner 类,它收集某些股票的每日报价,并返回该股票当日价格的跨度. 今天股票价格的跨度被定义为股票价格小于或等于今天价格的最大连续日数(从今天开始往回数,包括今天). 例如 ...
- [leetcode/lintcode 题解] Amazon面试题:连接棒材的最低费用
为了装修新房,你需要加工一些长度为正整数的棒材 sticks. 如果要将长度分别为 X 和 Y 的两根棒材连接在一起,你需要支付 X + Y 的费用. 由于施工需要,你必须将所有棒材连接成一根. 返回 ...
- [leetcode/lintcode 题解] Google面试题:合法组合
给一个单词s,和一个字符串集合str.这个单词每次去掉一个字母,直到剩下最后一个字母.求验证是否存在一种删除的顺序,这个顺序下所有的单词都在str中.例如单词是’abc’,字符串集合是{‘a’,’ab ...
- [leetcode/lintcode 题解] 微软面试题:公平索引
现在给你两个长度均为N的整数数组 A 和 B. 当(A[0]+...A[K-1]),(A[K]+...+A[N-1]),(B[0]+...+B[K-1]) 和 (B[K]+...+B[N-1])四个和 ...
- [leetcode/lintcode 题解] 微软 面试题:实现 Trie(前缀树)
实现一个 Trie,包含 insert, search, 和 startsWith 这三个方法. 在线评测地址:领扣题库官网 样例 1: 输入: insert(" ...
- Tree Operations 打印出有向图中的环
题目: You are given a binary tree with unique integer values on each node. However, the child pointers ...
- leetcode & lintcode 题解
刷题备忘录,for bug-free 招行面试题--求无序数组最长连续序列的长度,这里连续指的是值连续--间隔为1,并不是数值的位置连续 问题: 给出一个未排序的整数数组,找出最长的连续元素序列的长度 ...
- [LeetCode] Find All Numbers Disappeared in an Array 找出数组中所有消失的数字
Given an array of integers where 1 ≤ a[i] ≤ n (n = size of array), some elements appear twice and ot ...
- 一起来刷《剑指Offer》-- 题目一:找出数组中重复的数字(Python多种方法实现)
数组中重复的数字 最近在复习算法和数据结构(基于Python实现),然后看了Python的各种"序列"--比如列表List.元组Tuple和字符串String,后期会写一篇博客介绍 ...
随机推荐
- 入门大数据---Scala学习
Scala是什么? Scala是一种基于函数式编程和面向对象的高级语言.它开发了Spark等大型应用.它和Java有效集成,底层也是支持JVM的. 它有六大特性: 无缝JAVA互操作 Scala在JV ...
- 【转】HBase中Zookeeper,RegionServer,Master,Client之间关系
在2.0之前HDFS中只有一个NameNode,但对于在线的应用只有一个NameNode是不安全的,故在2.0中对NameNode进行抽象,抽象成NamService其下包含有多个NameNode,但 ...
- JavaScript基础JavaScript的常用编码惯例(007)
采用一定的编码惯例,可以使得项目中的代码提到较高的一致性,可读性和可预测性. 1.缩进缩 进可以提高代码的可读性.不过错误的缩进也可能导致代码的误读.有人认为缩进应该使用tab,另外的一些人主张采用4 ...
- Security 10:权限管理
SQL Server 用于管理权限的TSQL命令有:GRANT用于授予权限,REVOKE 用于移除授予的权限,而DENY用于防止安全主体通过GRANT获得权限.但是,SQL Server的权限管理不是 ...
- java简介&&变量
Day01 简介 数据 1.数据大体分为两类:基本类型和引用类型 2.基本类型的数据分为四类八种,四类为整型,浮点,布尔,字符 3.Long类型的数据超过int范围要在之后面加个L,不加L是整型会进行 ...
- 如何用好 IDEA ,Java 撸码效率至少提升 5 倍?
以前的Java项目中,充斥着太多不友好的代码:POJO的getter/setter/toString:异常处理:I/O流的关闭操作等等,这些样板代码既没有技术含量,又影响着代码的美观,Lombok应运 ...
- Solaris 10上Oracle 10g安装步骤图解
文章目录 1. 说明 2. 查看相关包 3. 添加用户和组 4. 设置oracle环境变量 5. 创建Oracle软件目录 6. 修改OS参数 7. 上传Oracle软件包并解压 8. 开始安装 9. ...
- 记一次解密wireshark抓取的冰蝎通信流量
一.关于冰蝎 1.1 简单介绍 冰蝎是一款基于Java开发的动态加密通信流量的新型Webshell客户端.老牌 Webshell 管理神器——中国菜刀的攻击流量特征明显,容易被各类安全设备检测,实际场 ...
- 看看有哪些 Web 认证技术.
BASIC 认证 BASIC 认证(基本认证)是从 HTTP/1.0 就定义的认证方式. BASIC 认证会将"用户名:密码"经过 Base64 加密后放入请求头部的 Author ...
- 常见Python的Web开发框架
在今天,Python里有很多开发框架用来帮助你轻松创建web应用.web开发框架存在的意义就在于可以快速便捷的构建应用,而不用去在意那些没必要的技术细节(协议.报文.数据结构). 到2020年为止,基 ...