代码随想录算法训练营第六十六天 |101.孤岛的总面积、102.沉没孤岛、103.水流问题、104.建造最大岛屿

本文主要是介绍代码随想录算法训练营第六十六天 |101.孤岛的总面积、102.沉没孤岛、103.水流问题、104.建造最大岛屿,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

101.孤岛的总面积

文字讲解:101. 孤岛的总面积 | 代码随想录

解题思路

本题要求找到不靠边的陆地面积,那么我们只要从周边找到陆地然后 通过 dfs或者bfs 将周边靠陆地且相邻的陆地都变成海洋,然后再去重新遍历地图 统计此时还剩下的陆地就可以了。

在遇到地图周边陆地的时候,将1都变为0,此时地图为这样:

 这里使用深搜

#include<bits/stdc++.h>
using namespace std;
int dir[4][2] = {0, 1, 1,0, 0,-1, -1,0};
int result =0 ;   //用于后续记录结果
void dfs(vector<vector<int>>& grid , int x, int y)
{grid[x][y] = 0;result++;int n = grid.size();int m = grid[0].size();for(int i=0 ; i<4 ; i++){int nextx = x + dir[i][0];int nexty = y + dir[i][1];if(nextx<0 || nextx>= n || nexty < 0 ||nexty >= m) continue;  //越界了if(grid[nextx][nexty]==0) continue;   //已经是海了dfs(grid,nextx,nexty);}return;
}int main()
{int n,m;cin >> n >> m;vector<vector<int>> grid(n,vector<int>(m,0));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++){if(grid[i][0]==1) dfs(grid,i,0);if(grid[i][m-1]==1) dfs(grid,i,m-1);}for(int j=0 ; j< m ; j++){if(grid[0][j]==1) dfs(grid,0,j);if(grid[n-1][j]==1) dfs(grid,n-1,j);}result = 0;   //全变为海洋后,再去遍历剩余的孤岛for(int i =0; i<n; i++){for(int j =0 ;j < m; j++){if(grid[i][j]==1) dfs(grid,i,j);}}cout << result << endl;
}

102.沉没孤岛

文字讲解:102. 沉没孤岛 | 代码随想录 

解题思路

步骤一:深搜或者广搜将地图周边的 1 (陆地)全部改成 2 (特殊标记)

步骤二:将水域中间 1 (陆地)全部改成 水域(0)

步骤三:将之前标记的 2 改为 1 (陆地)

这次我们使用广搜

#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 , int x, int y)
{queue<pair<int,int>> que;que.push({x,y});grid[x][y]=2;       //将连接的岛屿都变为2while(!que.empty()){pair<int,int> cur = que.front();que.pop();int curx = cur.first;int cury = cur.second;for(int i =0 ;i < 4; i++){int nextx = curx + dir[i][0];int nexty = cury + dir[i][1];int n = grid.size();int m = grid[0].size();if(nextx<0 || nextx>= n || nexty <0 || nexty >=m ) continue;if(grid[nextx][nexty]==0 || grid[nextx][nexty]==2) continue;//如果下一个格子是海洋或者已经标记过的,就跳过que.push({nextx,nexty});grid[nextx][nexty] = 2;}}
}int main()
{int n,m;cin >> n >> m;vector<vector<int>> grid(n,vector<int>(m,0));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++)   //标记两侧{if(grid[i][0]==1) bfs(grid,i,0);if(grid[i][m-1]==1) bfs(grid,i,m-1);}for(int j =0 ; j< m; j++)   //标记上下{if(grid[0][j]==1) bfs(grid,0,j);if(grid[n-1][j]==1) bfs(grid,n-1,j);}for(int i =0 ; i< n ; i++){for(int j = 0 ; j< m ;j++){if(grid[i][j]==1) grid[i][j] = 0;   //变为海洋if(grid[i][j]==2 ) grid[i][j] = 1;}}for(int i =0 ; i< n ; i++){for(int j = 0 ; j< m ;j++){cout<< grid[i][j] << " ";}cout << endl;}
}

103.水流问题

文字讲解:103. 水流问题 | 代码随想录

思路一:遍历每个点,去看是否能到达两个边界,但这样肯定超时,遍历每个节点是m*n的复杂度,而深搜每个节点又是m*n的复杂度

思路二:逆流优化

从第一组边界上的节点 逆流而上,将遍历过的节点都标记上。

同样从第二组边界的边上节点 逆流而上,将遍历过的节点也标记上。

然后两方都标记过的节点就是既可以流太平洋也可以流大西洋的节点

 

 这里使用深搜

#include<bits/stdc++.h>
using namespace std;
int dir[4][2] = {0,1,1,0,0,-1,-1,0};
void dfs(vector<vector<int>>& grid ,vector<vector<bool>>& visited , int x, int y)
{   if(visited[x][y])return;visited[x][y] = true;for(int i = 0 ; i< 4 ;i++){int nextx = x + dir[i][0];int nexty = y + dir[i][1];int n = grid.size();int m = grid[0].size();if(nextx<0 || nextx>= n || nexty<0 || nexty>=m) continue;  //越界if(grid[x][y] > grid[nextx][nexty]) continue;   //我们这里是从低到高遍历dfs(grid,visited,nextx,nexty);}return;
}int main()
{int n,m;cin>>n>>m;vector<vector<int>> grid(n,vector<int>(m,0));vector<vector<bool>> firstBoard(n,vector<bool>(m,0));vector<vector<bool>> secondBoard(n,vector<bool>(m,0));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++){dfs(grid,firstBoard,i,0);dfs(grid,secondBoard,i,m-1);}for(int j=0 ; j<m ; j++){dfs(grid,firstBoard,0,j);dfs(grid,secondBoard,n-1,j);}for(int i =0 ;i< n; i++){for(int j = 0; j<m ;j++){if(firstBoard[i][j]&&secondBoard[i][j]) cout<< i << " " << j <<endl;}}
}

104.建造最大岛屿

文字讲解: 104.建造最大岛屿 | 代码随想录

解题思路

思路一:计算地图的最大面积:遍历地图 + 深搜岛屿,时间复杂度为 n * n。 

每改变一个0的方格,都需要重新计算一个地图的最大面积,所以 整体时间复杂度为:n^4

 

思路二:

其实每次深搜遍历计算最大岛屿面积,我们都做了很多重复的工作。

只要用一次深搜把每个岛屿的面积记录下来就好。

第一步:一次遍历地图,得出各个岛屿的面积,并做编号记录。可以使用map记录,key为岛屿编号,value为岛屿面积

第二步:再遍历地图,遍历0的方格(因为要将0变成1),并统计该1(由0变成的1)周边岛屿面积,将其相邻面积相加在一起,遍历所有 0 之后,就可以得出 选一个0变成1 之后的最大面积。

拿如下地图的岛屿情况来举例: (1为陆地)

 

第二步:再遍历地图,遍历每一个0的位置,求最大面积即可 

 时间复杂度:O(n^2)

 

#include<bits/stdc++.h>
using namespace std;
int n,m;
int dir[4][2] = {0,1,1,0,0,-1,-1,0};
int area = 0; //面积
void dfs(vector<vector<int>>& grid, vector<vector<bool>>& visited, int x, int y, int mark) {if (visited[x][y] || grid[x][y] == 0) return; // 终止条件:访问过的节点 或者 遇到海水visited[x][y] = true; // 标记访问过grid[x][y] = mark; // 给陆地标记新标签area++;for (int i = 0; i < 4; i++) {int nextx = x + dir[i][0];int nexty = y + dir[i][1];if (nextx < 0 || nextx >= n || nexty < 0 || nexty >= m) continue;  // 越界了,直接跳过dfs(grid, visited, nextx, nexty, mark);}
}int main()
{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];}}int mask =2;   //从2开始标记unordered_map<int,int> map;   //用来存面积bool allGrid = true;for(int i=0 ;i < n ; i++){for(int j=0; j < m ; j++){if(grid[i][j] == 0) allGrid = false;if(!visited[i][j] && grid[i][j]==1)   //遇到辛岛屿了{area = 0;  //面积重置为0dfs(grid,visited,i,j,mask);map[ mask ] = area;   //记录这个岛屿的面积mask++;}}}if(allGrid){cout << n * m <<endl;return 0;}int result = 0;unordered_set<int> visitedGrid;for(int i=0 ;i < n ; i++){for(int j=0; j < m ; j++){area = 1;   //变为1后的岛屿总面积visitedGrid.clear();   //每次换格子变1时需要重置if(grid[i][j]==0){for (int k = 0; k < 4; k++) {int neari = i + dir[k][1]; // 计算相邻坐标int nearj = j + dir[k][0];if (neari < 0 || neari >= n || nearj < 0 || nearj >= m) continue;if (visitedGrid.count(grid[neari][nearj])) continue; // 添加过的岛屿不要重复添加// 把相邻四面的岛屿数量加起来area += map[grid[neari][nearj]];visitedGrid.insert(grid[neari][nearj]); // 标记该岛屿已经添加过}}result = max(result,area);}}cout << result << endl;
}

这篇关于代码随想录算法训练营第六十六天 |101.孤岛的总面积、102.沉没孤岛、103.水流问题、104.建造最大岛屿的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1081620

相关文章

linux生产者,消费者问题

pthread_cond_wait() :用于阻塞当前线程,等待别的线程使用pthread_cond_signal()或pthread_cond_broadcast来唤醒它。 pthread_cond_wait() 必须与pthread_mutex 配套使用。pthread_cond_wait()函数一进入wait状态就会自动release mutex。当其他线程通过pthread

问题:第一次世界大战的起止时间是 #其他#学习方法#微信

问题:第一次世界大战的起止时间是 A.1913 ~1918 年 B.1913 ~1918 年 C.1914 ~1918 年 D.1914 ~1919 年 参考答案如图所示

随想录 Day 69 并查集 107. 寻找存在的路径

随想录 Day 69 并查集 107. 寻找存在的路径 理论基础 int n = 1005; // n根据题目中节点数量而定,一般比节点数量大一点就好vector<int> father = vector<int> (n, 0); // C++里的一种数组结构// 并查集初始化void init() {for (int i = 0; i < n; ++i) {father[i] = i;}

uniapp接入微信小程序原生代码配置方案(优化版)

uniapp项目需要把微信小程序原生语法的功能代码嵌套过来,无需把原生代码转换为uniapp,可以配置拷贝的方式集成过来 1、拷贝代码包到src目录 2、vue.config.js中配置原生代码包直接拷贝到编译目录中 3、pages.json中配置分包目录,原生入口组件的路径 4、manifest.json中配置分包,使用原生组件 5、需要把原生代码包里的页面修改成组件的方

2024.6.24 IDEA中文乱码问题(服务器 控制台 TOMcat)实测已解决

1.问题产生原因: 1.文件编码不一致:如果文件的编码方式与IDEA设置的编码方式不一致,就会产生乱码。确保文件和IDEA使用相同的编码,通常是UTF-8。2.IDEA设置问题:检查IDEA的全局编码设置和项目编码设置是否正确。3.终端或控制台编码问题:如果你在终端或控制台看到乱码,可能是终端的编码设置问题。确保终端使用的是支持你的文件的编码方式。 2.解决方案: 1.File -> S

公共筛选组件(二次封装antd)支持代码提示

如果项目是基于antd组件库为基础搭建,可使用此公共筛选组件 使用到的库 npm i antdnpm i lodash-esnpm i @types/lodash-es -D /components/CommonSearch index.tsx import React from 'react';import { Button, Card, Form } from 'antd'

vcpkg安装opencv中的特殊问题记录(无法找到opencv_corexd.dll)

我是按照网上的vcpkg安装opencv方法进行的(比如这篇:从0开始在visual studio上安装opencv(超详细,针对小白)),但是中间出现了一些别人没有遇到的问题,虽然原因没有找到,但是本人给出一些暂时的解决办法: 问题1: 我在安装库命令行使用的是 .\vcpkg.exe install opencv 我的电脑是x64,vcpkg在这条命令后默认下载的也是opencv2:x6

17.用300行代码手写初体验Spring V1.0版本

1.1.课程目标 1、了解看源码最有效的方式,先猜测后验证,不要一开始就去调试代码。 2、浓缩就是精华,用 300行最简洁的代码 提炼Spring的基本设计思想。 3、掌握Spring框架的基本脉络。 1.2.内容定位 1、 具有1年以上的SpringMVC使用经验。 2、 希望深入了解Spring源码的人群,对 Spring有一个整体的宏观感受。 3、 全程手写实现SpringM

问题-windows-VPN不正确关闭导致网页打不开

为什么会发生这类事情呢? 主要原因是关机之前vpn没有关掉导致的。 至于为什么没关掉vpn会导致网页打不开,我猜测是因为vpn建立的链接没被更改。 正确关掉vpn的时候,会把ip链接断掉,如果你不正确关掉,ip链接没有断掉,此时你vpn又是没启动的,没有域名解析,所以就打不开网站。 你可以在打不开网页的时候,把vpn打开,你会发现网络又可以登录了。 方法一 注意:方法一虽然方便,但是可能会有

代码随想录算法训练营:12/60

非科班学习算法day12 | LeetCode150:逆波兰表达式 ,Leetcode239: 滑动窗口最大值  目录 介绍 一、基础概念补充: 1.c++字符串转为数字 1. std::stoi, std::stol, std::stoll, std::stoul, std::stoull(最常用) 2. std::stringstream 3. std::atoi, std