《LeetCode 200 FloodFill 岛屿数量DFS解法》

一.题目

200. 岛屿数量 - 力扣(LeetCode)

二.思路讲解

2.1 思路讲解

FloodFill(洪水灌溉) 系列的题目,同样可以采取深搜的方式处理,但与常规回溯算法有一个最显著的区别:不需要恢复现场 。回溯中的"恢复现场"是为了尝试其他分支,而 FloodFill 是一次性将整个连通区域填满 ,走过的格子被永久修改(例如将 '1' 改为 '0'),因此不需要再撤销操作。这就像真正的洪水灌溉,水流经过的地方就被覆盖了,不会再变回原样。

三.代码演示

cpp 复制代码
class Solution 
{
public:
    int ret;
    int row,col;
    bool check[301][301];
    int bx[4] = {0,0,1,-1};
    int by[4] = {1,-1,0,0};
    int numIslands(vector<vector<char>>& grid) 
    {
        //初始化
        row = grid.size(),col = grid[0].size();
        for(int i = 0;i < row;i++)
        {
            for(int j = 0;j < col;j++)
            {
                if(grid[i][j] == '1' && !check[i][j])
                {
                    dfs(grid,i,j);
                    ret++;
                }
            }
        }    
        return ret;
    }
    void dfs(vector<vector<char>>& grid,int i,int j)
    {
        for(int k = 0;k < 4;k++)
        {
            int x = i + bx[k],y = j + by[k];
            if(x >= 0 && y >= 0 && x < row && y < col && !check[x][y] && grid[x][y] == '1')
            {
                check[x][y] = true;//占用
                dfs(grid,x,y);
            }
        }
    }
};

四.代码讲解

一、全局变量与数据结构
  • ret:整型变量,记录岛屿数量,初始为 0。

  • rowcol:成员变量,存储网格的行数和列数。

  • check[301][301]:布尔数组,记录每个格子是否已被访问过,避免重复遍历。大小 301 足以应对常见数据范围。

  • 方向数组 bx[4]by[4]:表示上下左右四个方向的偏移量,用于探索相邻格子。

二、主函数 numIslands
  1. 获取网格的行数 row 和列数 col

  2. 遍历整个网格,对于每个格子 (i, j)

    • 如果当前格子是陆地 '1' 且未被访问过(!check[i][j]),则说明发现了一个新的岛屿。

    • 调用递归函数 dfs 从该格子开始进行深度优先搜索,将整个岛屿的陆地全部标记为已访问。

    • 岛屿计数 ret 加 1。

  3. 遍历结束后,返回岛屿数量 ret

三、递归函数 dfs

dfs(grid, i, j) 的作用是:从当前格子 (i, j) 出发,向四个方向探索,将所有与之相连的陆地格子标记为已访问。注意:当前格子本身也需要标记,否则会导致重复访问和死循环。但原代码在调用前并未标记起点,这是一个潜在错误,应修正。正确流程如下:

1. 标记当前格子

首先将当前格子标记为已访问(check[i][j] = true)。这是洪水灌溉的关键,保证每个格子只被处理一次。

2. 向四个方向探索

使用 for 循环遍历四个方向,计算新坐标 (x, y)。对于每个方向,检查该格子是否合法:

  • 边界检查x >= 0 && y >= 0 && x < row && y < col

  • 未访问!check[x][y]

  • 是陆地grid[x][y] == '1'

如果条件满足,则递归调用 dfs(grid, x, y),继续标记相邻的陆地。

四、关键细节
  • 无需回溯:FloodFill 与回溯不同,格子一旦被标记为已访问,就永久改变状态,不需要撤销。因此递归返回后无需恢复现场。

  • 访问标记的时机 :必须在进入递归前将当前格子标记为已访问,否则会导致重复访问,甚至无限递归。原代码在 dfs 内部未标记起点,这是一个错误。修正方法是在 dfs 开头添加 check[i][j] = true;

五、流程图讲解

相关推荐
星恒随风1 小时前
C++ string 入门(一)
开发语言·c++·笔记·学习
拂拉氏1 小时前
【知识讲解-题目讲解】算法系列之动态规划入门(下)
算法·leetcode·动态规划
skywalk81631 小时前
继续推进心语项目6.15 @CodeArts
开发语言·算法·编程
2601_961845151 小时前
花生十三图推思维导图|图形推理|技巧
数据结构·算法·链表·贪心算法·排序算法·线性回归·动态规划
前进吧-程序员1 小时前
反转链表完全指南:辅助容器、三指针、头插法
数据结构·c++·链表
我不是懒洋洋1 小时前
从零实现一个分布式配置中心:服务发现与热更新
c++
bIo7lyA8v1 小时前
算法复杂度的可视化评估与优化策略研究的技术8
算法
省四收割者1 小时前
从硬件中断到分布式协程:全景解构高并发机制与 C / Golang 的巅峰对决
c++·分布式·嵌入式硬件·golang
Cx330❀1 小时前
【Linux网络】从零定制应用层协议:黏包问题、全双工缓冲区与 Jsoncpp 序列化深度解析
linux·运维·服务器·开发语言·网络·c++·人工智能