代码随想录算法训练营第四十四天 | 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)
		}
	}
}
相关推荐
地平线开发者15 小时前
SparseDrive 模型导出与性能优化实战
算法·自动驾驶
董董灿是个攻城狮15 小时前
大模型连载2:初步认识 tokenizer 的过程
算法
地平线开发者15 小时前
地平线 VP 接口工程实践(一):hbVPRoiResize 接口功能、使用约束与典型问题总结
算法·自动驾驶
罗西的思考16 小时前
AI Agent框架探秘:拆解 OpenHands(10)--- Runtime
人工智能·算法·机器学习
HXhlx19 小时前
CART决策树基本原理
算法·机器学习
Wect19 小时前
LeetCode 210. 课程表 II 题解:Kahn算法+DFS 双解法精讲
前端·算法·typescript
颜酱20 小时前
单调队列:滑动窗口极值问题的最优解(通用模板版)
javascript·后端·算法
Gorway1 天前
解析残差网络 (ResNet)
算法
拖拉斯旋风1 天前
LeetCode 经典算法题解析:优先队列与广度优先搜索的巧妙应用
算法
Wect1 天前
LeetCode 207. 课程表:两种解法(BFS+DFS)详细解析
前端·算法·typescript