任务日期: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).