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

avatar
作者
筋斗云
阅读量:0

任务日期:8.2

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

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

代码:

#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:内圈一圈一圈地遍历),然后将符合条件的节点压入队列中并标记,然后在弹出队列中的元素一圈一圈的遍历循环往复一直到队列为空。

代码:

#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一圈一圈地遍历,这里包括外圈和内圈。

广告一刻

为您即时展示最新活动产品广告消息,让您随时掌握产品活动新动态!