Leetcode: Number of Islands II && Summary of Union Find
A 2d grid map of m rows and n columns is initially filled with water. We may perform an addLand operation which turns the water at position (row, col) into a land. Given a list of positions to operate, count the number of islands after each addLand operation. An island is surrounded by water and is formed by connecting adjacent lands horizontally or vertically. You may assume all four edges of the grid are all surrounded by water. Example: Given m = 3, n = 3, positions = [[0,0], [0,1], [1,2], [2,1]].
Initially, the 2d grid grid is filled with water. (Assume 0 represents water and 1 represents land). 0 0 0
0 0 0
0 0 0
Operation #1: addLand(0, 0) turns the water at grid[0][0] into a land. 1 0 0
0 0 0 Number of islands = 1
0 0 0
Operation #2: addLand(0, 1) turns the water at grid[0][1] into a land. 1 1 0
0 0 0 Number of islands = 1
0 0 0
Operation #3: addLand(1, 2) turns the water at grid[1][2] into a land. 1 1 0
0 0 1 Number of islands = 2
0 0 0
Operation #4: addLand(2, 1) turns the water at grid[2][1] into a land. 1 1 0
0 0 1 Number of islands = 3
0 1 0
We return the result as an array: [1, 1, 2, 3] Challenge: Can you do it in time complexity O(k log mn), where k is the length of the positions?
Union Find
Princeton's lecture note on Union Find in Algorithms and Data Structures It is a well organized note with clear illustration describing from the naive QuickFind to the one with Weighting and Path compression. With Weighting and Path compression, The algorithm runs in
O((M+N) log* N) where M is the number of operations ( unite and find ), N is the number of objects, log* is iterated logarithm while the naive runs in O(MN).
方法一: Union Find based on Quick Find
我觉得:Union复杂度: O(M*N), where M is the number of calls of Union, and N is the size of id array, in our case N=m*n
Find复杂度: O(1)
实际运行时间199ms
public class Solution {
public List<Integer> numIslands2(int m, int n, int[][] positions) {
int[][] dirs = new int[][]{{-1,0},{1,0},{0,1},{0,-1}};
unionFind uf = new unionFind(m*n);
List<Integer> res = new ArrayList<Integer>();
for (int[] pos : positions) {
int cur = pos[0]*n + pos[1];
uf.ids[cur] = cur;
uf.count++;
for (int[] dir : dirs) {
int x = dir[0] + pos[0];
int y = dir[1] + pos[1];
int nb = x*n+y;
if (x<0 || x>=m || y<0 || y>=n || uf.ids[nb]==-1) continue;
if (uf.find(nb) != uf.find(cur)) {
uf.union(nb, cur);
}
}
res.add(uf.count);
}
return res;
}
public class unionFind {
int[] ids;
int count;
public unionFind(int num) {
this.ids = new int[num];
Arrays.fill(ids, -1);
this.count = 0;
}
public int find(int num) {
return ids[num];
}
public boolean union(int n1, int n2) {
int id1=ids[n1], id2=ids[n2];
if (id1 != id2) {
for (int i=0; i<ids.length; i++) {
if (ids[i] == id2) {
ids[i] = id1;
}
}
count--;
return true;
}
return false;
}
}
}
Faster Union Find方法2:Union Find Based on Quick Union 参考:https://leetcode.com/discuss/69572/easiest-java-solution-with-explanations
Quick Union is Faster than Quick Find
The idea is simple. To represent a list of islands, we use trees. i.e., a list of roots. This helps us find the identifier of an island faster. If roots[c] = p means the parent of node c is p, we can climb up the parent chain to find out the identifier of an island, i.e., which island this point belongs to:
Do root[root[roots[c]]]... until root[c] == c;
To transform the two dimension problem into the classic UF, perform a linear mapping:
int id = n * x + y;
Initially assume every cell are in non-island set {-1}. When point A is added, we create a new root, i.e., a new island. Then, check if any of its 4 neighbors belong to the same island. If not,union the neighbor by setting the root to be the same. Remember to skip non-island cells.
我觉得:Union复杂度: O(M*logN), where M is the number of calls of Union, and N is the size of id array, in our case N=m*n
Find复杂度: O(logN)
实际运行28ms
public class Solution {
public List<Integer> numIslands2(int m, int n, int[][] positions) {
int[][] dirs = new int[][]{{-1,0},{1,0},{0,1},{0,-1}};
unionFind uf = new unionFind(m*n);
List<Integer> res = new ArrayList<Integer>();
for (int[] pos : positions) {
int cur = pos[0]*n + pos[1];
uf.ids[cur] = cur;
uf.count++;
for (int[] dir : dirs) {
int x = dir[0] + pos[0];
int y = dir[1] + pos[1];
int nb = x*n+y;
if (x<0 || x>=m || y<0 || y>=n || uf.ids[nb]==-1) continue;
int rootNb = uf.root(nb);
int rootCur = uf.root(cur);
if (rootCur != rootNb) { //not connect
uf.union(rootCur, rootNb);
uf.count--;
}
}
res.add(uf.count);
}
return res;
}
public class unionFind { //ids[]记录上一跳pos,root记录最上面的pos,union(i, j)修改i的root的上一跳为j的root
int[] ids;
int count;
public unionFind(int num) {
this.ids = new int[num];
Arrays.fill(ids, -1);
this.count = 0;
}
public int root(int i) { //FIND operation is proportional to the depth of the tree.the average running time is O(logN)
while (ids[i] != i) i = ids[i];
return i;
}
public boolean isConnected(int i, int j) {
return root(i) == root(j);
}
public void union(int i, int j) {
int iRoot = root(i);
int jRoot = root(j);
ids[iRoot] = jRoot;
}
}
}
Summary of Union Find:
Princeton's lecture note on Union Find
Quick Find

Quick Union
Here is a very easy understanding video by Stanford(看3:00开始的例子,非常简单, 一看就懂)

Compare of Fast Find & Fast Union, though worst case time complexity is almost the same, fast union is faster than fast find

Leetcode: Number of Islands II && Summary of Union Find的更多相关文章
- [LeetCode] Number of Islands II 岛屿的数量之二
A 2d grid map of m rows and n columns is initially filled with water. We may perform an addLand oper ...
- LeetCode – Number of Islands II
A 2d grid map of m rows and n columns is initially filled with water. We may perform an addLand oper ...
- [LeetCode] Number of Islands II
Problem Description: A 2d grid map of m rows and n columns is initially filled with water. We may pe ...
- [LeetCode] Number of Islands 岛屿的数量
Given a 2d grid map of '1's (land) and '0's (water), count the number of islands. An island is surro ...
- [LeetCode] 305. Number of Islands II 岛屿的数量之二
A 2d grid map of m rows and n columns is initially filled with water. We may perform an addLand oper ...
- LeetCode 305. Number of Islands II
原题链接在这里:https://leetcode.com/problems/number-of-islands-ii/ 题目: A 2d grid map of m rows and n column ...
- [LeetCode] 305. Number of Islands II 岛屿的数量 II
A 2d grid map of m rows and n columns is initially filled with water. We may perform an addLand oper ...
- 305. Number of Islands II
题目: A 2d grid map of m rows and n columns is initially filled with water. We may perform an addLand ...
- [Swift]LeetCode305. 岛屿的个数 II $ Number of Islands II
A 2d grid map of m rows and n columns is initially filled with water. We may perform an addLand oper ...
随机推荐
- 5 个 Composer 小技巧
Composer是新一代的PHP依赖管理工具.其介绍和基本用法可以看这篇<Composer PHP依赖管理的新时代>.本文介绍使用Composer的五个小技巧,希望能给你的PHP开发带来方 ...
- 【转】NGUI创建UIRoot后报NullReferenceException的解决办法
本文参考自 http://forum.china.unity3d.com/thread-1099-1-1.html 使用NGUI版本3.7.5. 在创建了一个UIRoot后,有时会报NullRefer ...
- Python中dict的特点、更新dict、遍历dict
dict的第一个特点是查找速度快,无论dict有10个元素还是10万个元素,查找速度都一样.而list的查找速度随着元素增加而逐渐下降. 不过dict的查找速度快不是没有代价的,dict的缺点是占用内 ...
- yii多数据库
Yii中同时连接多个数据库 Published by 荒野无灯 on 2011-07-09 02:12:45 under PHP/Yii Tags:yii,database 14162 views 0 ...
- 1106 C程序语法树
- mongodb在win7下的安装和使用
1.下载mongodb的windows版本,有32位和64位版本,根据系统情况下载,下载地址:http://www.mongodb.org/downloads 2.解压缩至额E:/mongodb即可 ...
- Android Studio工具修理集
本文来自http://blog.csdn.net/liuxian13183/ ,引用必须注明出处! 1.Common依赖项目找不到.因为主项目没有引进setting.gradle 2.从Eclipse ...
- raspberryPi 拍照
调用python的库,学习raspberryPi的摄像头操作方法. 参考链接: https://www.raspberrypi.org/learning/getting-started-with-pi ...
- table tricks
- 学习WEb前端开发的需要哪些条件
第一阶段--HTML的学习. 超文本标记语言(HyperText Mark-up Language 简称HTML)是一个网页的骨架,无论是静态网页还是动态网页,最终返回到浏览器端的都是HTML代码,浏 ...