算法打卡day2 (2026-02-07 周五) | 算法: DFS | 3_卡码网99_计数孤岛_DFS

- 第 162 篇 -
Date: 2026 - 02- 07
Author: 郑龙浩(仟墨)

2026-02-07 周五 | 算法打卡day2

文章目录

3_卡码网99_计数孤岛_DFS

算法: DFS

题目描述

给定一个由 1(陆地)和 0(水)组成的矩阵,你需要计算岛屿的数量。岛屿由水平方向或垂直方向上相邻的陆地连接而成,并且四周都是水域。你可以假设矩阵外均被水包围。

输入描述

第一行包含两个整数 N, M,表示矩阵的行数和列数。

后续 N 行,每行包含 M 个数字,数字为 1 或者 0。

输出描述

输出一个整数,表示岛屿的数量。如果不存在岛屿,则输出 0。

输入示例

复制代码
4 5
1 1 0 0 0
1 1 0 0 0
0 0 1 0 0
0 0 0 1 1

输出示例

复制代码
3
cpp 复制代码
// Date: 2026-02-07 Author:郑龙浩

// 3_卡码99_计数孤岛_DFS.cpp

// DFS

// 用时: 60min

#include "bits/stdc++.h"

using namespace std;

  

int direction[4][2] = {{0, 1}, {0, -1}, {1, 0}, {-1, 0}};

  

// 写法1:显性终止条件(先判断坐标是否合理,后标记,再dfs搜索下一个坐标)

void dfs1(vector <vector <int>>& grid, vector <vector <bool>>& visited, int x, int y) {

    int height = grid.size(); // 高

    int width = grid[0].size(); // 宽

    if (x < 0 ||  x > height - 1 || y < 0 || y > width - 1)  return; // 如果越界,终止

    if (grid[x][y] == 0 || visited[x][y] == true) return; // 如果当前坐标 是海洋,或者访问过,终止

    visited[x][y] = true; // 将当前坐标标记为 已访问过

    // 只检查周围四个坐标点

    int nextX, nextY;

    for (int i = 0; i < 4; i ++) {

        nextX = x + direction[i][0];

        nextY = y + direction[i][1];

        dfs1(grid, visited, nextX, nextY); // 意思:用dfs深搜遍历 (nextX, nextY) 周围的四个点

    }

}

  

// 写法2:隐性终止条件(先标记,后面检查坐标,如果不合条件,是不会执行下一次的标记的,所以前面可以在不检查的条件下进行标记的操作)

void dfs2(vector <vector <int>>& grid, vector <vector <bool>>& visited, int x, int y) {

    int height = grid.size(); // 高

    int width = grid[0].size(); // 宽

  

    visited[x][y] = true; // 将当前坐标标记为 已访问过

  

    // 只检查周围四个坐标点

    int nextX, nextY;

    for (int i = 0; i < 4; i ++) {

        nextX = x + direction[i][0];

        nextY = y + direction[i][1];

        if (nextX < height && nextX >= 0 && nextY < width && nextY >= 0 && grid[nextX][nextY] == 1 && visited[nextX][nextY] == false) { // 如果下一个坐标没有越界且是陆地,且没有访问过,就用dfs深搜遍历它周围四个点

            dfs2(grid, visited, nextX, nextY);

        }

    }

}

  

int main(void) {

    ios::sync_with_stdio(0);

    cin.tie(0); cout.tie(0);

    int N, M, cnt = 0;

    cin >> N >> M;

    vector <vector <int>> grid(N, vector<int>(M, 0)); // 存储矩阵

    vector <vector <bool>> visited(N, vector<bool>(M, false)); // 记录每个位置是否被访问过

  

    for(int i = 0; i < N; i++) {

        for (int j = 0; j < M; j++) {

            cin >> grid[i][j];

        }

    }

  

    for (int i = 0; i < N; i++) {

        for (int j = 0; j < M; j++) {

            // 如果当前位置是陆地且未被访问,说明发现了一个新岛屿

            if (grid[i][j] == 1 && visited[i][j] == 0) {

                dfs2(grid, visited, i, j); // 通过DFS标记这个岛屿的所有相连陆地

                cnt ++; // 岛屿数量加1

            }

        }

    }

    cout << cnt;

    return 0;

}
相关推荐
吃好睡好便好2 小时前
在Matlab中绘制横直方图
开发语言·学习·算法·matlab
仰泳之鹅2 小时前
【C语言】自定义数据类型2——联合体与枚举
c语言·开发语言·算法
x_yeyue5 小时前
三角形数
笔记·算法·数论·组合数学
念何架构之路6 小时前
Go语言加密算法
数据结构·算法·哈希算法
AI科技星6 小时前
《数学公理体系·第三部·数术几何》(2026 年版)
c语言·开发语言·线性代数·算法·矩阵·量子计算·agi
失去的青春---夕阳下的奔跑6 小时前
560. 和为 K 的子数组
数据结构·算法·leetcode
黎阳之光6 小时前
黎阳之光:以视频孪生重构智慧医院信息化,打造高标项目核心竞争力
大数据·人工智能·物联网·算法·数字孪生
丷丩7 小时前
三级缓存下MVT地图瓦片服务性能优化策略
算法·缓存·性能优化·gis·geoai-up
m0_629494737 小时前
LeetCode 热题 100-----25.回文链表
数据结构·算法·leetcode·链表
ʚ希希ɞ ྀ8 小时前
单词拆分----dp
算法