LeetCode 刷题 [C++] 第200题. 岛屿数量 (广度优先遍历BFS与深度优先遍历DFS)

题目描述

给你一个由 ‘1’(陆地)和 ‘0’(水)组成的的二维网格,请你计算网格中岛屿的数量。
岛屿总是被水包围,并且每座岛屿只能由水平方向和/或竖直方向上相邻的陆地连接形成。
此外,你可以假设该网格的四条边均被水包围。

示例 1:
输入:grid = [
  ["1","1","1","1","0"],
  ["1","1","0","1","0"],
  ["1","1","0","0","0"],
  ["0","0","0","0","0"]
]
输出:1

示例 2:
输入:grid = [
  ["1","1","0","0","0"],
  ["1","1","0","0","0"],
  ["0","0","1","0","0"],
  ["0","0","0","1","1"]
]
输出:3

提示:
    m == grid.length
    n == grid[i].length
    1 <= m, n <= 300
    grid[i][j] 的值为 '0' 或 '1'

广度优先遍历 BFS

题目是找到矩阵中 “岛屿的数量” ,因此上下左右相连的’1’都被认为是同一个连续的岛屿。
1.主循环:遍历整个矩阵,当遇到 grid[i][j] == ‘1’ 时,从此点开始做深度优先搜索 dfs,岛屿数 count + 1 且在深度优先搜索中删除此岛屿。
2.bfs方法:借用一个队列queue,将未越界且为字符’1’的节点 (i, j)加入队列:

  • 弹出队列首节点,并将此节点上下左右节点 (i+1,j),(i-1,j),(i,j+1),(i,j-1)中未越界且为字符’1’的加入队列;
  • 若其上下左右节点不符合条件,则不进行任何操作,继续弹出队列首元素节点;
  • 循环pop队列首节点,直到整个队列为空,此时已经遍历完当前岛屿。

具体实现代码如下

class Solution {
public:
    void bfs(vector<vector<char>>& grid,int i,int j,int cow,int col) {
        queue<vector<int>> curQue;
        curQue.emplace(vector<int>{i,j});
        while(!curQue.empty()) {
            auto curNode = curQue.front();
            curQue.pop();
            int m=curNode[0],n=curNode[1];
            if(m-1>=0 && grid[m-1][n] == '1') {
                grid[m-1][n] = '0';
                curQue.emplace(vector<int>{m-1,n});
            }
            if(m+1<cow && grid[m+1][n] == '1') {
                grid[m+1][n] = '0';
                curQue.emplace(vector<int>{m+1,n});
            }
            if(n-1>=0 && grid[m][n-1] == '1') {
                grid[m][n-1] = '0';
                curQue.emplace(vector<int>{m,n-1});
            }
            if(n+1<col && grid[m][n+1] == '1') {
                grid[m][n+1] = '0';
                curQue.emplace(vector<int>{m,n+1});
            }
        }
    }
    int numIslands(vector<vector<char>>& grid) {
        int count=0,cow=grid.size(),col=grid[0].size();
        for(int i=0;i<cow;++i) {
            for(int j=0;j<col;++j) {
                if('1' == grid[i][j]) {
                    grid[i][j] = '0';
                    bfs(grid,i,j,cow,col);
                    ++count;
                }
            }
        }
        return count;
    }
};

AC结果
在这里插入图片描述

深度优先遍历DFS

1.主循环:遍历整个矩阵,当遇到 grid[i][j] == ‘1’ 时,从此点开始做深度优先搜索 dfs,岛屿数 count + 1 且在深度优先搜索中删除此岛屿。
2.dfs方法: 设目前指针指向一个岛屿中的某一点 (i, j),寻找包括此点的岛屿边界。

  • 从 (i, j) 向此点的上下左右 (i+1,j),(i-1,j),(i,j+1),(i,j-1) 做深度搜索。
  • 终止条件:(i, j) 越过矩阵边界; grid[i][j] == ‘0’,代表此分支已越过岛屿边界。
  • 搜索岛屿的同时,执行 grid[i][j] = ‘0’,即将岛屿所有节点删除,以免之后重复搜索相同岛屿。

具体实现代码如下

class Solution {
public:
    void dfs(vector<vector<char>>& grid,int i,int j) {
        if(i<0 || i>=grid.size() || j<0 || j>=grid[i].size() || grid[i][j]=='0' ) return;
        grid[i][j]='0';
        dfs(grid,i,j-1);
        dfs(grid,i,j+1);
        dfs(grid,i-1,j);
        dfs(grid,i+1,j);
    }
    int numIslands(vector<vector<char>>& grid) {
        int nr = grid.size();
        if (nr == 0) return 0;
        int nc = grid[0].size();

        int count=0;
        for(int i=0;i<nr;++i) {
            for(int j=0;j<nc;++j) {
                if('1' == grid[i][j]) {
                    dfs(grid,i,j);
                    ++count;
                }
            }
        }
        return count;
    }
};

AC结果
在这里插入图片描述

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
引用中的代码片段是一个递归函数,名为beautifulArray,它接受一个整数n作为参数,并返回一个vector<int>类型的结果。这个函数用于生成一个长度为n的漂亮数组,漂亮数组满足以下条件:如果将其划分为两个任意长度的子数组,那么两个子数组的元素之和不相等。 函数的实现思路是通过递归,将问划分为两个子问,然后合并子问的解。首先判断n是否等于1,如果等于1,则直接返回一个包含1的数组。否则,将n减1后的一半传入递归函数beautifulArray,得到一个漂亮数组res1。然后将n的一半传入递归函数beautifulArray,得到另一个漂亮数组res2。最后,将res1中的每个元素乘以2并减去1,然后与res2中的每个元素乘以2合并到结果数组res中。 引用中的代码片段是一个函数,名为translateNum,它接受一个整数num作为参数,并返回一个整数。这个函数用于计算将整数num翻译成字符串的不同方法数。 函数的实现思路是将整数num转换为字符串str,然后创建一个长度为str.size()-1的动态数组dp,用于记录不同位置的翻译方法数。然后初始化dp和dp[1]为1,表示前两位数字的翻译方法数。接下来,从第三位数字开始遍字符串str,如果当前数字与前两位数字可以构成一个在10到25之间的整数,则将dp[i]的值设置为dp[i-1]+dp[i-2],表示当前位置的翻译方法数为前一位和前两位数字的翻译方法数之和。最后,返回dp最后一个元素的值,即为翻译整数num的不同方法数。 引用中的代码片段是一个函数,名为numberOfArithmeticSlices,它接受一个vector<int>类型的数组nums作为参数,并返回一个整数。这个函数用于计算数组nums中等差子序列的个数。 函数的实现思路是通过动态规划,创建一个与nums长度相同的动态数组dp,用于记录以每个位置为结尾的等差子序列的个数。然后遍数组nums,从第三个元素开始,判断当前元素与前两个元素是否构成等差数列,如果是,则将dp[i]的值设置为dp[i-1]加1,表示以当前位置为结尾的等差子序列个数为前一位的等差子序列个数加1。最后,返回dp数组中所有元素的和,即为等差子序列的个数。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值