zoukankan      html  css  js  c++  java
  • LeetCode 200. 岛屿数量

    我的LeetCode:https://leetcode-cn.com/u/ituring/

    我的LeetCode刷题源码[GitHub]:https://github.com/izhoujie/Algorithmcii

    LeetCode 200. 岛屿数量

    题目

    给你一个由 '1'(陆地)和 '0'(水)组成的的二维网格,请你计算网格中岛屿的数量。

    岛屿总是被水包围,并且每座岛屿只能由水平方向和/或竖直方向上相邻的陆地连接形成。

    此外,你可以假设该网格的四条边均被水包围。

    示例 1:

    输入:
    11110
    11010
    11000
    00000
    输出: 1
    

    示例 2:

    输入:
    11000
    11000
    00100
    00011
    输出: 3
    解释: 每座岛屿只能由水平和/或竖直方向上相邻的陆地连接而成。
    

    来源:力扣(LeetCode)
    链接:https://leetcode-cn.com/problems/number-of-islands
    著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

    解题思路

    比较自然能想到的是用dfs解决,且比较方便,也可以用bfs,每个1都需要一次bfs;

    思路1-dfs给岛屿编号

    每遇到一个1就dfs,并给从'2'开始的编号,遍历完最后的编号减去'2'即岛屿数;

    算法复杂度:

    • M为行数,N为列数;
    • 时间复杂度: $ {color{Magenta}{Omicronleft(M ast N ight)}} $
    • 空间复杂度: $ {color{Magenta}{Omicronleft(M ast N ight)}} $

    思路2-bfs给岛屿编号

    每遇到一个1就bfs,并给从'2'开始的编号,遍历完最后的编号减去'2'即岛屿数;

    小技巧:以前bfs遇到数组这样的数据,在构造队列存储时需要用数组存二维坐标,但是本题把坐标转化为int存储

    • row为行数,col为列数,对于坐标[i,j],t=i*col+j,存t;
    • 取的时候因为j<col,所以i=t/col,j=t%col,这样把二维压缩为一维就避免了创建数组这个比较耗时的操作;

    算法复杂度:

    • M为行数,N为列数;
    • 时间复杂度: $ {color{Magenta}{Omicronleft(M ast N ight)}} $
    • 空间复杂度: $ {color{Magenta}{Omicronleft(Minleft(M,N ight) ight)}} $

    算法源码示例

    package leetcode;
    
    import java.util.Deque;
    import java.util.LinkedList;
    
    /**
     * @author ZhouJie
     * @date 2020年4月20日 下午3:23:26 
     * @Description: 200. 岛屿数量
     *
     */
    public class LeetCode_0200 {
    
    }
    
    class Solution_0200 {
    	/**
    	 * @author: ZhouJie
    	 * @date: 2020年4月20日 下午4:12:48 
    	 * @param: @param grid
    	 * @param: @return
    	 * @return: int
    	 * @Description: 1-DFS,每找到一个1就给岛屿一个编号并DFS,最后命名的数量就是岛屿数;
    	 *
    	 */
    	public int numIslands_1(char[][] grid) {
    		if (grid == null || grid.length == 0 || grid[0].length == 0) {
    			return 0;
    		}
    		int row = grid.length;
    		int col = grid[0].length;
    		// 因为源数组是'1',所以岛屿编号从'2'开始
    		char count = '2';
    		for (int i = 0; i < row; i++) {
    			for (int j = 0; j < col; j++) {
    				// 每找到一个'1'就dfs并标记 为当前编号
    				if (grid[i][j] == '1') {
    					dfs(grid, i, j, count);
    					// 标记完当前岛屿编号递增
    					count++;
    				}
    			}
    		}
    		// 一共标记的岛屿数
    		return count - '2';
    	}
    
    	/**
    	 * @author: ZhouJie
    	 * @date: 2020年4月20日 下午4:13:59 
    	 * @param: @param grid
    	 * @param: @param i
    	 * @param: @param j
    	 * @param: @param count
    	 * @return: void
    	 * @Description: dfs
    	 *
    	 */
    	private void dfs(char[][] grid, int i, int j, char count) {
    		if (i >= 0 && i < grid.length && j >= 0 && j < grid[0].length && grid[i][j] == '1') {
    			grid[i][j] = count;
    			dfs(grid, i - 1, j, count);
    			dfs(grid, i + 1, j, count);
    			dfs(grid, i, j - 1, count);
    			dfs(grid, i, j + 1, count);
    		}
    	}
    
    	/**
    	 * @author: ZhouJie
    	 * @date: 2020年4月20日 下午4:22:47 
    	 * @param: @param grid
    	 * @param: @return
    	 * @return: int
    	 * @Description: 2-BFS,不同与同源BFS,本题恰是要计算不同源的数量,所以每遇到一个'1'就需要一次BFS; 
    	 *
    	 */
    	public int numIslands_2(char[][] grid) {
    		if (grid == null || grid.length == 0 || grid[0].length == 0) {
    			return 0;
    		}
    		int row = grid.length;
    		int col = grid[0].length;
    		// 因为源数组是'1',所以岛屿编号从'2'开始
    		char count = '2';
    		Deque<Integer> queue = new LinkedList<Integer>();
    		for (int i = 0; i < row; i++) {
    			for (int j = 0; j < col; j++) {
    				// 每找到一个'1'就dfs并标记 为当前编号
    				if (grid[i][j] == '1') {
    					// 这里存ij的位置很巧妙,用col乘以i再加j,因为j<col,所以取回时除以col就能再得到i,而对col取模就得到了j,避免了用数组存ij
    					queue.offer(i * col + j);
    					while (!queue.isEmpty()) {
    						Integer poll = queue.poll();
    						int x = poll / col;
    						int y = poll % col;
    						if (x > 0 && grid[x - 1][y] == '1') {
    							grid[x - 1][y] = count;
    							queue.offer((x - 1) * col + y);
    						}
    						if (y > 0 && grid[x][y - 1] == '1') {
    							grid[x][y - 1] = count;
    							queue.offer(x * col + y - 1);
    						}
    						if (x < row - 1 && grid[x + 1][y] == '1') {
    							grid[x + 1][y] = count;
    							queue.offer((x + 1) * col + y);
    						}
    						if (y < col - 1 && grid[x][y + 1] == '1') {
    							grid[x][y + 1] = count;
    							queue.offer(x * col + y + 1);
    						}
    
    					}
    					// 标记完当前岛屿编号递增
    					count++;
    				}
    			}
    		}
    		return count - '2';
    	}
    }
    
  • 相关阅读:
    strcmp()比较函数和strcasecmp()和strnatcmp()
    substr()函数
    改变字符串中的字母大小写
    explode()与相反函数 implode() 和join()
    PHP nl2br() 函数
    PHP trim() 函数
    PHP的count(数组)和strlen(字符串)的内部实现
    变量处理函数库
    php中定义数组的方法
    80端口的烦恼:[3]清除NT Kernel占用80端口
  • 原文地址:https://www.cnblogs.com/izhoujie/p/12740362.html
Copyright © 2011-2022 走看看