200. 岛屿的个数

https://leetcode-cn.com/problems/number-of-islands/

解法一:dfs

思路很直接,遍历棋盘,遇到‘1’就dfs,把访问过的地方置‘0’。统计dfs次数即可

class Solution:
    def numIslands(self, grid: List[List[str]]) -> int:
        count = 0  #统计岛屿数
        #遍历棋盘
        for i in range(len(grid)):
            for j in range(len(grid[0])):
                #遇到‘1’就调用递归dfs搜索,岛屿数+1
                if grid[i][j] == '1':  
                    count += 1
                    self.dfs(grid, i, j)
                    #递归完再加也可
                    #count += 1
        return count
    
    #深度优先遍历所有‘1’,并置零
    def dfs(self, grid, i, j):
        #特别要注意边界,除了越界,还有‘0’也要返回,否则会无限重复访问
        if i < 0 or i >= len(grid) or j < 0 or j >= len(grid[0]) or grid[i][j] == '0':
            return
        grid[i][j] = '0'    #访问该岛屿,并标记为0
        #上下左右
        self.dfs(grid, i-1, j)
        self.dfs(grid, i+1, j)
        self.dfs(grid, i, j-1)
        self.dfs(grid, i, j+1)

二、并查集

并查集初始化为m*n规模(每个点都是一个根),此处略作修改,连通分量个数为‘1’的数量,‘0’不计入。

遍历二维网格,遇到‘1’就尝试将上下左右的1连通,最后统计连通分量数

class UnionFind:
    # 构造函数传入totalNodes为总节点数
    def __init__(self, totalNodes, count):
        # parents[i]表示i的根,初始i的根为其自身
        self.parents = [i for i in range(totalNodes)]
        # 连通分量数
        self.count = count
        # 树的“重量”
        self.size = [1] * totalNodes
    # 合并连通区域是通过find来操作的, 即看这两个节点是不是在一个连通区域内.
    def union(self, node1, node2):
        root1 = self.find(node1)
        root2 = self.find(node2)
        # 不连通才合并
        if root1 != root2:
          	# 小树合并到大树
            if self.size[root1] > self.size[root2]:
                self.parents[root2] = root1
                self.size[root1] += self.size[root2]
            else:
                self.parents[root1] = root2
                self.size[root2] += self.size[root1]
            # 连通分量数减一
            self.count -= 1

    # 查找最终的根
    def find(self, node):
        while self.parents[node] != node:
            self.parents[node] = self.parents[self.parents[node]]
            node = self.parents[node]
        return node

    # 判断两个点是否连通
    def isConnected(self, node1, node2):
        return self.find(node1) == self.find(node2)

    # 返回连通分量个数
    def count(self):
        return self.count

class Solution:
    def numIslands(self, grid: List[List[str]]) -> int:
        m = len(grid)
        n = len(grid[0])
        #将二维坐标展平为一维
        def flat(x, y):
            return x * n + y
        count = 0
        for i in range(m):
            for j in range(n):
                if grid[i][j] == '1':
                    count += 1
        uf = UnionFind(m*n, count) #初始:每个'1'结点都是一个连通分量
        for i in range(m):
            for j in range(n):
                if grid[i][j] == '1':
                    # 将当前1置0,减少重复合并。不置0也没错,会稍慢
                    grid[i][j] = '0'
                    #看左右上下是否存在1
                    if j > 0 and grid[i][j-1] == '1':
                        uf.union(flat(i,j), flat(i, j-1))
                    if j < n-1 and grid[i][j+1] == '1':
                        uf.union(flat(i,j), flat(i, j+1))
                    if i > 0 and grid[i-1][j] == '1':
                        uf.union(flat(i,j), flat(i-1, j))
                    if i < m-1 and grid[i+1][j] == '1':
                        uf.union(flat(i,j), flat(i+1, j))
        return uf.count

优化

遍历时,若当前节点为1,使用后可以置0以减少重复计算

130. 被围绕的区域不同,那里求并查集之后还需要用到’O‘的位置信息,因此使用后不能置’X‘

最后更新于