代码随想录算法训练营第四十四天 | 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)
		}
	}
}
相关推荐
Yzzz-F17 小时前
Problem - 2205D - Codeforces
算法
智者知已应修善业18 小时前
【51单片机2个按键控制流水灯运行与暂停】2023-9-6
c++·经验分享·笔记·算法·51单片机
Halo_tjn18 小时前
Java Set集合相关知识点
java·开发语言·算法
生成论实验室18 小时前
《事件关系阴阳博弈动力学:识势应势之道》第四篇:降U动力学——认知确定度的自驱演化
人工智能·科技·神经网络·算法·架构
AI科技星19 小时前
全域数学·72分册:场计算机卷【乖乖数学】
算法·机器学习·数学建模·数据挖掘·量子计算
科研前沿19 小时前
镜像孪生VS视频孪生核心技术产品核心优势
大数据·人工智能·算法·重构·空间计算
水蓝烟雨19 小时前
1931. 用三种不同颜色为网格涂色
算法·leetcode
晨曦夜月20 小时前
map与unordered_map区别
算法·哈希算法
图码20 小时前
如何用多种方法判断字符串是否为回文?
开发语言·数据结构·c++·算法·阿里云·线性回归·数字雕刻
handler0120 小时前
Linux 内核剖析:进程优先级、上下文切换与 O(1) 调度算法
linux·运维·c语言·开发语言·c++·笔记·算法