代码随想录算法训练营第四十四天 | 99.岛屿数量 深搜 99.岛屿数量 广搜 100. 岛屿的最大面积

版本一的写法是 :下一个节点是否能合法已经判断完了,传进dfs函数的就是合法节点。

版本二的写法是:不管节点是否合法,上来就dfs,然后在终止条件的地方进行判断,不合法再return。

我在之前回溯算法做过笔记,我更偏好版本一。

xy老让我联想到坐标,我就不用xy了。也可以叫row、col。

Go 复制代码
package main

import (
	"bufio"
	"fmt"
	"os"
)

var dir = [4][2]int{{0, 1}, {1, 0}, {-1, 0}, {0, -1}}

func main() {
	in := bufio.NewReader(os.Stdin)
	var n, m int
	fmt.Fscan(in, &n, &m)
	grid := make([][]int, n)
	visited := make([][]bool, n)
	for i := range grid {
		grid[i] = make([]int, m)
	}
	for i := range visited {
		visited[i] = make([]bool, m)
	}
	for i := range grid {
		for j := range grid[i] {
			fmt.Fscan(in, &grid[i][j])
		}
	}
	res := 0
	for i := range grid {
		for j := range grid[i] {
			if grid[i][j] == 1 && !visited[i][j] {
				res++
				visited[i][j] = true
				dfs(grid, visited, i, j)
			}
		}
	}
	fmt.Println(res)
}
func dfs(grid [][]int, visited [][]bool, i, j int) {
	for k := 0; k < 4; k++ {
		nextI := i + dir[k][0]
		nextJ := j + dir[k][1]
		if nextI < 0 || nextI >= len(grid) || nextJ < 0 || nextJ >= len(grid[0]) {
			continue
		}
		if grid[nextI][nextJ] == 1 && !visited[nextI][nextJ] {
			visited[nextI][nextJ] = true
			dfs(grid, visited, nextI, nextJ)
		}
	}
}

如果节点出队列再标记为已访问过,会导致相同的节点重复入队列,进而导致队列中会有大量的重复节点。

Go 复制代码
package main

import (
	"bufio"
	"fmt"
	"os"
)

var dir = [4][2]int{{0, 1}, {1, 0}, {-1, 0}, {0, -1}}

func main() {
	in := bufio.NewReader(os.Stdin)
	var n, m int
	fmt.Fscan(in, &n, &m)
	grid := make([][]int, n)
	visited := make([][]bool, n)
	for i := range grid {
		grid[i] = make([]int, m)
	}
	for i := range visited {
		visited[i] = make([]bool, m)
	}
	for i := range grid {
		for j := range grid[i] {
			fmt.Fscan(in, &grid[i][j])
		}
	}
	res := 0
	for i := range grid {
		for j := range grid[i] {
			if grid[i][j] == 1 && !visited[i][j] {
				res++
				visited[i][j] = true
				bfs(grid, visited, i, j)
			}
		}
	}
	fmt.Println(res)
}

type Pair struct {
	i, j int
}

func bfs(grid [][]int, visited [][]bool, row, col int) {
	q := make([]Pair, 0)
	q = append(q, Pair{row, col})
	visited[row][col] = true

	for len(q) != 0 {
		cur := q[0]
		q = q[1:]
		for k := 0; k < 4; k++ {
			nextI := cur.i + dir[k][0]
			nextJ := cur.j + dir[k][1]
			if nextI < 0 || nextI >= len(grid) || nextJ < 0 || nextJ >= len(grid[0]) {
				continue
			}
			if grid[nextI][nextJ] == 1 && !visited[nextI][nextJ] {
				q = append(q, Pair{nextI, nextJ})
				visited[nextI][nextJ] = true
			}
		}
	}
}

easy

Go 复制代码
package main

import (
	"bufio"
	"fmt"
	"os"
)

var dir = [4][2]int{{0, 1}, {1, 0}, {-1, 0}, {0, -1}}
var count int

func main() {
	in := bufio.NewReader(os.Stdin)
	var n, m int
	fmt.Fscan(in, &n, &m)
	grid := make([][]int, n)
	visited := make([][]bool, n)
	for i := range grid {
		grid[i] = make([]int, m)
	}
	for i := range visited {
		visited[i] = make([]bool, m)
	}
	for i := range grid {
		for j := range grid[i] {
			fmt.Fscan(in, &grid[i][j])
		}
	}
	res := 0
	for i := range grid {
		for j := range grid[i] {
			if grid[i][j] == 1 && !visited[i][j] {
				visited[i][j] = true
				count = 1
				dfs(grid, visited, i, j)
				if count > res {
					res = count
				}
			}
		}
	}
	fmt.Println(res)
}
func dfs(grid [][]int, visited [][]bool, i, j int) {
	for k := 0; k < 4; k++ {
		nextI := i + dir[k][0]
		nextJ := j + dir[k][1]
		if nextI < 0 || nextI >= len(grid) || nextJ < 0 || nextJ >= len(grid[0]) {
			continue
		}
		if grid[nextI][nextJ] == 1 && !visited[nextI][nextJ] {
			visited[nextI][nextJ] = true
			count++
			dfs(grid, visited, nextI, nextJ)
		}
	}
}
相关推荐
JieE2124 小时前
LeetCode 101. 对称二叉树|JS 递归 + 迭代双解法,彻底搞懂镜像判断
javascript·算法
JieE2121 天前
LeetCode 56. 合并区间|超清晰 JS 图解思路,面试高频区间题
javascript·算法·面试
Jack201 天前
HarmonyOS开发中错误处理策略:网络异常统一处理
算法
小小杨树2 天前
读懂色彩:拍照调色不再难
算法·计算机视觉·配色
JieE2122 天前
LeetCode 226. 翻转二叉树|JS 递归超详细拆解,二叉树入门经典题
javascript·算法
JieE2122 天前
LeetCode 104. 二叉树的最大深度|递归思路超详细拆解
javascript·算法
vivo互联网技术2 天前
CVPR 2026 | 全新强化学习框架 BeautyGRPO:重塑真实人像
算法·大模型·cvpr·影像
Darling噜啦啦2 天前
列表转树算法深度解析:从 Map 到 Reduce 的两种实现,面试高频考点
数据结构·算法·面试
用户497863050733 天前
(一)小红的数组操作
算法·编程语言