代码随想录算法训练营第六十六天 |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

相关文章

IntelliJ IDEA 中配置 Spring MVC 环境的详细步骤及问题解决

《IntelliJIDEA中配置SpringMVC环境的详细步骤及问题解决》:本文主要介绍IntelliJIDEA中配置SpringMVC环境的详细步骤及问题解决,本文分步骤结合实例给大... 目录步骤 1:创建 Maven Web 项目步骤 2:添加 Spring MVC 依赖1、保存后执行2、将新的依赖

Spring 中的循环引用问题解决方法

《Spring中的循环引用问题解决方法》:本文主要介绍Spring中的循环引用问题解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录什么是循环引用?循环依赖三级缓存解决循环依赖二级缓存三级缓存本章来聊聊Spring 中的循环引用问题该如何解决。这里聊

Spring Boot中JSON数值溢出问题从报错到优雅解决办法

《SpringBoot中JSON数值溢出问题从报错到优雅解决办法》:本文主要介绍SpringBoot中JSON数值溢出问题从报错到优雅的解决办法,通过修改字段类型为Long、添加全局异常处理和... 目录一、问题背景:为什么我的接口突然报错了?二、为什么会发生这个错误?1. Java 数据类型的“容量”限制

利用Python调试串口的示例代码

《利用Python调试串口的示例代码》在嵌入式开发、物联网设备调试过程中,串口通信是最基础的调试手段本文将带你用Python+ttkbootstrap打造一款高颜值、多功能的串口调试助手,需要的可以了... 目录概述:为什么需要专业的串口调试工具项目架构设计1.1 技术栈选型1.2 关键类说明1.3 线程模

关于MongoDB图片URL存储异常问题以及解决

《关于MongoDB图片URL存储异常问题以及解决》:本文主要介绍关于MongoDB图片URL存储异常问题以及解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录MongoDB图片URL存储异常问题项目场景问题描述原因分析解决方案预防措施js总结MongoDB图

SpringBoot项目中报错The field screenShot exceeds its maximum permitted size of 1048576 bytes.的问题及解决

《SpringBoot项目中报错ThefieldscreenShotexceedsitsmaximumpermittedsizeof1048576bytes.的问题及解决》这篇文章... 目录项目场景问题描述原因分析解决方案总结项目场景javascript提示:项目相关背景:项目场景:基于Spring

Python Transformers库(NLP处理库)案例代码讲解

《PythonTransformers库(NLP处理库)案例代码讲解》本文介绍transformers库的全面讲解,包含基础知识、高级用法、案例代码及学习路径,内容经过组织,适合不同阶段的学习者,对... 目录一、基础知识1. Transformers 库简介2. 安装与环境配置3. 快速上手示例二、核心模

解决Maven项目idea找不到本地仓库jar包问题以及使用mvn install:install-file

《解决Maven项目idea找不到本地仓库jar包问题以及使用mvninstall:install-file》:本文主要介绍解决Maven项目idea找不到本地仓库jar包问题以及使用mvnin... 目录Maven项目idea找不到本地仓库jar包以及使用mvn install:install-file基

Java的栈与队列实现代码解析

《Java的栈与队列实现代码解析》栈是常见的线性数据结构,栈的特点是以先进后出的形式,后进先出,先进后出,分为栈底和栈顶,栈应用于内存的分配,表达式求值,存储临时的数据和方法的调用等,本文给大家介绍J... 目录栈的概念(Stack)栈的实现代码队列(Queue)模拟实现队列(双链表实现)循环队列(循环数组

usb接口驱动异常问题常用解决方案

《usb接口驱动异常问题常用解决方案》当遇到USB接口驱动异常时,可以通过多种方法来解决,其中主要就包括重装USB控制器、禁用USB选择性暂停设置、更新或安装新的主板驱动等... usb接口驱动异常怎么办,USB接口驱动异常是常见问题,通常由驱动损坏、系统更新冲突、硬件故障或电源管理设置导致。以下是常用解决