day 51 第十一章:图论part02 99.岛屿数量 深搜 99.岛屿数量 广搜 100.岛屿的最大面积

任务日期:8.2

题目一链接: 99. 岛屿数量 (kamacoder.com)

思路:主函数:将题目里的数据放入一个矩阵graph里,然后遍历graph,遇见没有遍历过的陆地就result + 1同时dfs当前节点。dfs函数:此题就一个递归终止条件所以不用写确定递归终止的条件,然后确定单层递归逻辑,分别向上右下左四个方向进行递归,符合条件的继续下一层递归。

代码:

cpp 复制代码
#include <bits/stdc++.h>
using namespace std;
//思路:用遇到一个没有遍历过的节点陆地,计数器就加一,然后把该节点陆地所能遍历到的陆地都标记上。
int dir[4][2] = {0,1,1,0,0,-1,-1,0};//上右下左
void dfs(const vector<vector<int>> &gragh,vector<vector<bool>> &visited,int x,int y) {
    //确定递归终止条件:不需要再写,因为终止条件在下面的if语句里了;一道题一般有两个终止条件,一个是这里一个在下面
    //if(visited[x][y] || gragh[x][y] == 0) return;//如果当前点被遍历过或者是海水,则终止递归
    //确定当前层递归逻辑:到下一层的过程
    for(int i = 0;i < 4;i ++) {//依次遍历四个位置
        int nextx = x + dir[i][0];//每个位置的第一个数控制x
        int nexty = y + dir[i][1];//每个位置的第一个数控制y
        if(nextx < 0 || nextx >= gragh.size() || nexty < 0 || nexty >= gragh[0].size()) continue;//如果越界则跳过这个点
        if(gragh[nextx][nexty] == 1 && !visited[nextx][nexty]) {//如果当前节点及时陆地又没被标记
            visited[nextx][nexty] = true;//先标记
            dfs(gragh,visited,nextx,nexty);//沿当前方向继续下一层递归
        }
    }
}

int main() {
    int n,m;
    cin>>n>>m;
    std::vector<vector<int>> gragh(n + 1,vector<int>(m + 1,0));
    //将题目输入数据记录到一个邻接矩阵中
    for(int i = 0;i < n;i ++) {
        for(int j = 0;j < m;j ++) {
            cin>>gragh[i][j];
        }
    }
    //定义一个记录已经被标记的陆地的邻接矩阵
    vector<vector<bool>> visited(n + 1,vector<bool>(m + 1,false));//visited数组的类型是bool并且初始化为false
    int result = 0;//记录结果
    //遍历邻接矩阵graph
    for(int i = 0;i < n;i ++) {//遍历整个gragh,寻找没有遍历过得陆地
        for(int j = 0;j < m;j ++) {
            if(gragh[i][j] == 1 && !visited[i][j]) {//遇到一个没有遍历过的陆地
                result ++;//计数器就加一
                visited[i][j] = true;//标记已经遍历的点
                dfs(gragh,visited,i,j);//然后把该节点陆地所能遍历到的陆地都标记上。
            }
        }
    }
    std::cout << result << std::endl;
    return 0;
}

难点:1.visited矩阵是bool类型,注意传递参数使得数据类型

2.一般dfs里面有两个终止递归的地方,一个在确定递归终止条件里,一个在确定单层递归逻辑里,如果当前限制条件较少,这两个终止条件只能写一个,否则答案会错误。

3.dir二维数组,弄清下标的意义。




题目二链接: 99. 岛屿数量 (kamacoder.com)

思路:遇到一个没有遍历过的节点陆地,计数器就加一,然后标记该节点陆地所能遍历到的所有陆地。主函数:将题目的数组放入到二维数组grid中,然后遍历grid数组中的每一个点,如果发现一个没有被标记的陆地则result + 1并且标记和bfs当前节点。bfs:同上省略确定递归终止条件的一步,在确定当前层函数时首先创建一个队列(用来实现外圈 一圈一圈地遍历),将当前点加入到队列中,然后遍历当前节点一周的节点(体现bfs:内圈一圈一圈地遍历),然后将符合条件的节点压入队列中并标记,然后在弹出队列中的元素一圈一圈的遍历循环往复一直到队列为空。

代码:

cpp 复制代码
#include <bits/stdc++.h>
using namespace std;
//思路:遇到一个没有遍历过的节点陆地,计数器就加一,然后把该节点陆地所能遍历到的陆地都标记上。
int dir[4][2] = {0,1,1,0,0,-1,-1,0};
void bfs(vector<vector<int>> &grid,vector<vector<bool>> &visited,int x,int y) {
    //确定递归终止条件:遍历的下一层是被标记的点或者是海洋
    //确定单层递归逻辑
    std::queue<pair<int,int>> que;
    que.push({x,y});//队列用push就行
    visited[x][y] = true;//压入到队列中就标记点x,y,否则某一点会重复加入进队列里
    while(!que.empty()) {
        pair<int,int> cur = que.front();
        que.pop();
        for(int i = 0;i < 4;i ++) {//遍历当前节点的四周一圈的节点
            int nextx = cur.first + dir[i][0];
            int nexty = cur.second + dir[i][1];
            if(nextx < 0 || nextx == grid.size() || nexty < 0 || nexty == grid[0].size()) continue;
            if(grid[nextx][nexty] == 1 && !visited[nextx][nexty]) {
                que.push({nextx,nexty});//四个方向的节点都加入到队列中,一圈一圈地遍历,同时还不用管顺序
                visited[nextx][nexty] = true;
            }
        }
    }
}

int main() {
    int n,m;
    cin>>n>>m;
    vector<vector<int>> grid(n + 1,vector<int>(m + 1,0));
    for(int i = 0;i < n;i ++) {
        for(int j = 0;j < m;j ++) {
            cin>>grid[i][j];
        }
    }
    int result = 0;
    vector<vector<bool>> visited(n + 1,vector<bool> (m + 1,false));
    //遍历grid
    for(int i = 0;i < n;i ++) {
        for(int j = 0;j < m;j ++) {
            if(grid[i][j] == 1 && !visited[i][j]) {
                result ++;
                visited[i][j] = true;
                bfs(grid,visited,i,j);
            }
        }
    }
    cout<<result;
    return 0;
}

难点:1.进入bfs后先创建一个队列并将当前节点加入队列同时标记该节点

2.在节点压入队列时就标记改点

3.队列控制外圈一圈一圈地遍历,for循环控制内圈一圈一圈地遍历。

4.此题不用管遍历时每一圈的顺序,所以用队列或者栈都行

5.bfs在形式上与dfs的区别就是bfs一圈一圈地遍历,这里包括外圈和内圈。




题目三:岛屿的最大面积

链接:100. 岛屿的最大面积 (kamacoder.com)

思路:遇到一个没有遍历过的节点陆地,计数器就加一,然后标记该节点陆地所能遍历到的所有陆地。

代码:

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
int dir[4][2] = {0,1,1,0,0,-1,-1,0};
int bfs(vector<vector<int>> &grid,vector<vector<bool>> &visited,int x,int y) {//在bfs里面定义一个count然后返回count
    std::queue<pair<int,int>> que;
    que.push({x,y});
    int count = 1;//初始值是1
    while(!que.empty()) {//外圈
        pair<int,int> cur;
        cur = que.front();//进行外圈的循环
        que.pop();
        for(int i = 0;i < 4;i ++) {//内圈循环
            int nextx = cur.first + dir[i][0];
            int nexty = cur.second + dir[i][1];
            if(nextx < 0 || nextx == grid.size() || nexty < 0 || nexty == grid[0].size()) continue;//判断是否越界
            if(grid[nextx][nexty] == 1 && !visited[nextx][nexty]) {//寻找没有被标记的陆地
                que.push({nextx,nexty});//为标记它周围的陆地做准备
                visited[nextx][nexty] = true;//标记当前陆地
                count ++;
            }
        }
    }
    return count;
}

int main() {
    int n,m;
    cin>>n>>m;
    std::vector<vector<int>> grid(n + 1,vector<int> (m + 1,0));
    for(int i = 0;i < n;i ++) {
        for(int j = 0;j < m;j ++) {
            cin>>grid[i][j];
        }
    }
    vector<vector<bool>> visited(n + 1,vector<bool> (m + 1,false));
    int result = 0;
    for(int i = 0;i < n;i ++) {
        for(int j = 0;j < m;j ++) {
            if(grid[i][j] == 1 && !visited[i][j]) {
                visited[i][j] = true;//标记当前的点
                result = max(result,bfs(grid,visited,i,j));//遍历他周围的点
            }
        }
    }
    cout<<result;
}

难点:

1.设置队列和相应的队列代码的作用:不理解的地方想本题的思路,由此可知它是用来标记为标记的陆地周围的为标记的陆地。

2.这道题的bfs的数据类型是int,在bfs里面定义一个count,最后返回到result = max(result,bfs).

相关推荐
飞升不如收破烂~35 分钟前
redis的map底层数据结构 分别什么时候使用哈希表(Hash Table)和压缩列表(ZipList)
算法·哈希算法
九圣残炎40 分钟前
【从零开始的LeetCode-算法】3354. 使数组元素等于零
java·算法·leetcode
程序猿小柒1 小时前
leetcode hot100【LeetCode 4.寻找两个正序数组的中位数】java实现
java·算法·leetcode
雨中rain2 小时前
贪心算法(1)
算法·贪心算法
不爱学习的YY酱2 小时前
【操作系统不挂科】<CPU调度(13)>选择题(带答案与解析)
java·linux·前端·算法·操作系统
平头哥在等你2 小时前
求一个3*3矩阵对角线元素之和
c语言·算法·矩阵
飞滕人生TYF2 小时前
动态规划 详解
算法·动态规划
_OLi_2 小时前
力扣 LeetCode 106. 从中序与后序遍历序列构造二叉树(Day9:二叉树)
数据结构·算法·leetcode
ahadee3 小时前
蓝桥杯每日真题 - 第18天
c语言·vscode·算法·蓝桥杯
地平线开发者3 小时前
CPU& 内存加压工具 stress-ng 介绍
算法·自动驾驶