图论day57|建造最大岛屿(卡码网)【截至目前所做的题中,图论的最高难度】
- 思维导图分析
- 104.建造最大岛屿(卡码网)【截至目前所做的题中,图论的最高难度】
思维导图分析
104.建造最大岛屿(卡码网)【截至目前所做的题中,图论的最高难度】
题目描述
给定一个由 1(陆地)和 0(水)组成的矩阵,你最多可以将矩阵中的一格水变为一块陆地,在执行了此操作之后,矩阵中最大的岛屿面积是多少。
岛屿面积的计算方式为组成岛屿的陆地的总数。岛屿是被水包围,并且通过水平方向或垂直方向上相邻的陆地连接而成的。你可以假设矩阵外均被水包围。
输入描述
第一行包含两个整数 N, M,表示矩阵的行数和列数。之后 N 行,每行包含 M 个数字,数字为 1 或者 0,表示岛屿的单元格。
输出描述
输出一个整数,表示最大的岛屿面积。
输入示例
4 5
1 1 0 0 0
1 1 0 0 0
0 0 1 0 0
0 0 0 1 1
输出示例
6
提示信息
对于上面的案例,有两个位置可将 0 变成 1,使得岛屿的面积最大,即 6。
数据范围:
1 <= M, N <= 50。
#include <iostream>
#include <vector>
#include <unordered_map>
#include <unordered_set>
using namespace std;int dir[4][2]={1,0,-1,0,0,1,0,-1};
int count;void dfs(vector<vector<int>> &grid,vector<vector<bool>>& visited,int x,int y,int mark)
{if(visited[x][y]==true||grid[x][y]==0)return;visited[x][y]=true;grid[x][y]=mark;count++;for(int i=0;i<4;i++){int nextx=x+dir[i][0];int nexty=y+dir[i][1];if(nextx<=0||nextx>=grid.size()||nexty<=0||nexty>=grid[1].size())continue;dfs(grid,visited,nextx,nexty,mark);}
}int main()
{int n,m;cin>>n>>m;vector<vector<int>> grid(n+1,vector<int>(m+1,0));for(int i=1;i<=n;i++)for(int j=1;j<=m;j++)cin>>grid[i][j];vector<vector<bool>> visited(n+1,vector<bool>(m+1,false));unordered_map<int,int> map;bool landAll=true;int mark=2;//给岛屿标记并用哈希表存储for(int i=1;i<=n;i++)for(int j=1;j<=m;j++){if(grid[i][j]==0)landAll=false;if(grid[i][j]==1&&visited[i][j]==false){count=0;bfs(grid,visited,i,j,mark);map[mark]=count;mark++;}}if(landAll){cout<<n*m<<endl;return 0;}unordered_set<int> set;int result=0;//将一块水变成陆地for(int i=1;i<=n;i++)for(int j=1;j<=m;j++){if(grid[i][j]==0){set.clear();count=1;for(int k=0;k<4;k++){int neari=i+dir[k][0];int nearj=j+dir[k][1];if(neari<=0||neari>=grid.size()||nearj<=0||nearj>=grid[1].size())continue;if(set.find(grid[neari][nearj])!=set.end())continue;count+=map[grid[neari][nearj]];set.insert(grid[neari][nearj]);}result=max(result,count);}}cout<<result<<endl;
}
具体分析过程见思维导图