【LeetCode 热题100】BFS/DFS 实战:岛屿数量 & 腐烂的橘子(力扣200 / 994 )(Go语言版)

简介: 本文讲解了两道经典的图论问题:**岛屿数量(LeetCode 200)** 和 **腐烂的橘子(LeetCode 994)**,分别通过 DFS/BFS 实现。在“岛屿数量”中,利用深度或广度优先搜索遍历二维网格,标记连通陆地并计数;“腐烂的橘子”则采用多源 BFS,模拟腐烂传播过程,计算最短时间。两者均需掌握访问标记技巧,是学习网格搜索算法的绝佳实践。

🌊 BFS/DFS 实战:岛屿数量 & 腐烂的橘子(LeetCode 200 & 994)

两道图论基础题,涉及 BFS 与 DFS 的应用,主要用于掌握二维网格中遍历与标记访问的技巧:

  • 🏝️ 200. 岛屿数量(Number of Islands)
  • 🍊 994. 腐烂的橘子(Rotting Oranges)

🏝️ 一、200. 岛屿数量

📌 题目描述

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

岛屿总是由相邻的陆地连接组成(水平或垂直),并且被水包围。


💡 解题思路

这道题的核心是:对每一个未被访问的陆地进行一次 DFS 或 BFS,将整块陆地标记为已访问,岛屿数量 +1

✅ 方法一:DFS 深度优先遍历

  • 从每个为 '1' 的位置出发,递归淹没它相邻的陆地;
  • 每次新的 DFS 开始,即发现了一个新的岛屿。
func numIslands(grid [][]byte) int {
   
    m, n := len(grid), len(grid[0])
    var dfs func(int, int)
    dfs = func(i, j int) {
   
        if i < 0 || j < 0 || i >= m || j >= n || grid[i][j] == '0' {
   
            return
        }
        grid[i][j] = '0'
        dfs(i+1, j)
        dfs(i-1, j)
        dfs(i, j+1)
        dfs(i, j-1)
    }

    count := 0
    for i := 0; i < m; i++ {
   
        for j := 0; j < n; j++ {
   
            if grid[i][j] == '1' {
   
                dfs(i, j)
                count++
            }
        }
    }
    return count
}

✅ 方法二:BFS 广度优先遍历

  • 使用队列,每次将 '1' 入队后,通过四个方向将邻接陆地逐个加入队列。

🧠 小结

技术点 说明
核心思想 将一整块陆地用 DFS/BFS 访问并标记
遍历方式 DFS / BFS 均可
时间复杂度 O(m × n)
空间复杂度 DFS:O(m × n)(递归栈),BFS:队列空间

🍊 二、994. 腐烂的橘子

📌 题目描述

在一个二维网格中:

  • 0 代表空单元格,
  • 1 代表新鲜橘子,
  • 2 代表腐烂橘子。

每分钟,所有腐烂橘子会让上下左右相邻的新鲜橘子变腐烂
求腐烂完所有橘子的最短分钟数,如果无法全部腐烂,返回 -1


💡 解题思路

这是标准的多源 BFS 问题,初始队列中包含所有腐烂橘子的位置,然后每轮向周围传播。

✅ 步骤总结:

  1. 初始化队列,将所有腐烂橘子的坐标加入;
  2. 记录新鲜橘子的数量 fresh
  3. 每轮扩散一层(即一分钟),对新鲜橘子变腐烂,fresh--
  4. 最终看 fresh 是否为 0,若是返回分钟数,否则返回 -1。

📦 Go 实现

type pair struct{
    x, y int }

func orangesRotting(grid [][]int) int {
   
    m, n := len(grid), len(grid[0])
    queue := []pair{
   }
    fresh := 0

    for i := 0; i < m; i++ {
   
        for j := 0; j < n; j++ {
   
            if grid[i][j] == 2 {
   
                queue = append(queue, pair{
   i, j})
            } else if grid[i][j] == 1 {
   
                fresh++
            }
        }
    }

    dirs := []pair{
   {
   0, 1}, {
   1, 0}, {
   0, -1}, {
   -1, 0}}
    minutes := 0

    for len(queue) > 0 && fresh > 0 {
   
        size := len(queue)
        for i := 0; i < size; i++ {
   
            curr := queue[0]
            queue = queue[1:]
            for _, d := range dirs {
   
                x, y := curr.x + d.x, curr.y + d.y
                if x >= 0 && y >= 0 && x < m && y < n && grid[x][y] == 1 {
   
                    grid[x][y] = 2
                    fresh--
                    queue = append(queue, pair{
   x, y})
                }
            }
        }
        minutes++
    }

    if fresh > 0 {
   
        return -1
    }
    return minutes
}

⚠️ 注意事项

  • 初始化时要统计新鲜橘子的数量
  • 每一轮是批量扩散(多源 BFS)
  • 不能单独计数某个腐烂橘子的时间,要整体一起推进。

🔚 总结对比

特性 岛屿数量(200) 腐烂的橘子(994)
模型 连通块计数 最短传播路径,多源 BFS
算法 DFS / BFS BFS(层级扩散)
是否需要标记访问
特别关注 连通性与计数 初始状态和时间控制

目录
相关文章
|
16天前
|
JSON 中间件 Go
Go语言实战指南 —— Go中的反射机制:reflect 包使用
Go语言中的反射机制通过`reflect`包实现,允许程序在运行时动态检查变量类型、获取或设置值、调用方法等。它适用于初中级开发者深入理解Go的动态能力,帮助构建通用工具、中间件和ORM系统等。
156 63
|
6天前
|
自然语言处理 算法 前端开发
Go语言实战案例-判断回文字符串-是不是正着念反着念都一样?
本案例实现判断回文字符串程序,支持中文、英文、标点及空格处理。通过双指针法与Unicode字符比较,帮助读者掌握字符串处理技巧与逻辑编程基础。
|
3天前
|
数据采集 机器学习/深度学习 存储
Go语言实战案例 - 找出切片中的最大值与最小值
本案例通过实现查找整数切片中的最大值与最小值,帮助初学者掌握遍历、比较和错误处理技巧,内容涵盖算法基础、应用场景及完整代码示例,适合初学者提升编程能力。
|
9天前
|
存储 小程序 Go
Go语言实战案例-读取用户输入并打印
本案例介绍如何使用Go语言的`fmt`包读取终端输入并输出信息。通过编写简单的交互程序,学习`fmt.Scanln()`、`fmt.Print()`和字符串拼接等基础I/O操作,适合初学者掌握命令行交互编程。
|
7天前
|
存储 算法 Go
Go语言实战案例-字符串反转
本案例通过“字符串反转”任务,帮助初学者理解Go语言中字符串的本质、Unicode字符处理、切片操作及基本算法思想(如双指针法)。内容涵盖字符串反转的多种应用场景,如回文判断、加密解密等,并提供完整代码实现与解析,支持中文及特殊字符处理,避免乱码问题。同时介绍了错误示范与进阶优化方法,如封装成函数及泛型版本,适合拓展练习与深入学习。
|
8天前
|
Go
Go语言实战案例-简易计算器(加减乘除)
本案例旨在实现一个控制台版简易计算器程序,能够读取用户输入的两个数字和一个运算符(+、-、*、/),并输出相应的计算结果。通过该案例,初学者可以练习基本的输入处理、条件判断、运算操作以及错误处理等编程技能。
|
1月前
|
Go
【LeetCode 热题100】DP 实战进阶:最长递增子序列、乘积最大子数组、分割等和子集(力扣300 / 152/ 416 )(Go语言版)
本文深入解析三道经典的动态规划问题:**最长递增子序列(LIS)**、**乘积最大子数组** 和 **分割等和子集**。 - **300. LIS** 通过 `dp[i]` 表示以第 `i` 个元素结尾的最长递增子序列长度,支持 O(n²) 动态规划与 O(n log n) 的二分优化。 - **152. 乘积最大子数组** 利用正负数特性,同时维护最大值与最小值的状态转移方程。 - **416. 分割等和子集** 转化为 0-1 背包问题,通过布尔型 DP 实现子集和判断。 总结对比了三题的状态定义与解法技巧,并延伸至相关变种问题,助你掌握动态规划的核心思想与灵活应用!
80 1
|
10月前
|
Unix Shell Linux
LeetCode刷题 Shell编程四则 | 194. 转置文件 192. 统计词频 193. 有效电话号码 195. 第十行
本文提供了几个Linux shell脚本编程问题的解决方案,包括转置文件内容、统计词频、验证有效电话号码和提取文件的第十行,每个问题都给出了至少一种实现方法。
144 6
LeetCode刷题 Shell编程四则 | 194. 转置文件 192. 统计词频 193. 有效电话号码 195. 第十行
|
11月前
|
Python
【Leetcode刷题Python】剑指 Offer 32 - III. 从上到下打印二叉树 III
本文介绍了两种Python实现方法,用于按照之字形顺序打印二叉树的层次遍历结果,实现了在奇数层正序、偶数层反序打印节点的功能。
116 6
|
11月前
|
搜索推荐 索引 Python
【Leetcode刷题Python】牛客. 数组中未出现的最小正整数
本文介绍了牛客网题目"数组中未出现的最小正整数"的解法,提供了一种满足O(n)时间复杂度和O(1)空间复杂度要求的原地排序算法,并给出了Python实现代码。
234 2