蓝桥杯2023(十四届)省赛——岛屿个数(DFS+BFS)

岛屿个数(DFS+BFS)

1.岛屿个数 - 蓝桥云课 (lanqiao.cn)

**法一:**本题很妙啊,你发现没有。反向操作,不是直接算岛屿数量,而是先将最外层的海洋给标注出来(DFS,visit[i][j]=2),然后再用一次DFS把所有visit[i][j]=0的当做整体,这些不管里面有没有子岛屿,有多少个子岛屿,都是一个大岛屿。

本题你可以学到的:标记海洋(八方、只遍历最外面一圈)、标记岛屿(四方、无回溯(因为我们要全部标记,不是找路径))

**法二:**这里提供一个思路,在评论区看到另一个大佬写的:搜索出所有岛屿,这个不难做到。由于岛屿之间互相隔离,则如果岛屿的一个格子在一个环内,那么整个岛屿也都在环内。遍历所有的岛屿,选中当前岛屿的第一个格子,搜索周围海洋,若能搜索到地图的边界外,则此岛屿不在任何一个环内;否则,此岛屿在某个环内,岛屿数量减一。

这个思路和我做的方法区别在于,我是从外向内,而该思路是正向思考,然后利用内岛的性质,对整体做出进一步加工。

下面给出法一的代码:

AC:DFS

#include <iostream>
#include <cstring>
#include <string>
#include <vector>
using namespace std;
//牛的,我是真的没想到,可以反向操作,不好求里面,那就求外面,把外层海水搜素出来,
// 这样剩下的不管有没有內岛屿,都是一个岛屿	他们用的是bfs,我这里用dfs试试,应该也可以
//注意,检索海水和检索岛屿不一样,海水是八方搜索,而岛屿是四方搜素const int N = 105;
int t, n, m;	//m行 n列
int nx[4] = { 0,1,0,-1 };
int ny[4] = { 1,0,-1,0 };
int seax[8] = { 0,1,1,1,0,-1,-1,-1 };
int seay[8] = { 1,1,0,-1,-1,-1,0,1 };
vector<string>g;
int visit[N][N] = { 0 };
int landNum = 0;void dfs(int x, int y)
{for (int i = 0; i < 8; i++){int xx = x + seax[i];int yy = y + seay[i];if (xx < 0 || xx >= m || yy < 0 || yy >= n)	continue;	//越界if (g[xx][yy] == '0' && visit[xx][yy] == 0){visit[xx][yy] = 2;dfs(xx, yy);//visit[xx][yy] = 0;	//回溯}}
}void dfsLand(int x, int y)
{for (int i = 0; i < 4; i++){int xx = x + nx[i];int yy = y + ny[i];if (xx < 0 || xx >= m || yy < 0 || yy >= n)	continue;	//越界if (visit[xx][yy] == 0)	//接下来就和g的值无关了{visit[xx][yy] = 1;dfsLand(xx, yy);}}
}int main()
{cin >> t;while (t--){g.clear();memset(visit, 0, sizeof(visit));landNum = 0;cin >> m >> n;getchar();string str;for (int i = 0; i < m; i++){getline(cin, str);g.push_back(str);}//检索海洋//for (int i = 0; i < n; i++)//{//	for (int j = 0; j < m; j++)//	{//		if (g[i][j] == '0' && visit[i][j] == 0)	//		{//			visit[i][j] = 2;	//最外层海水 标记为2//			dfs(i,j);//		}//	}//}		//这样标记海水不是最外层的,而是所有//要标记外海,其实只需要遍历最外面的一圈for (int i = 0; i < m; i++){for (int j = 0; j < n; j++){if (i == 0 || j == 0 || i == m - 1 || j == n - 1){if (g[i][j] == '0' && visit[i][j] == 0){visit[i][j] = 2;	//最外层海水 标记为2dfs(i, j);}}}}//检索岛屿for (int i = 0; i < m; i++){for (int j = 0; j < n; j++){if (visit[i][j] == 0)	//剩下的都是岛屿{landNum++;visit[i][j] = 1;	//陆地 标记为1dfsLand(i, j);}}}cout << landNum << endl;}return 0;
}

二刷:思路和一刷差不多,这里用的BFS,主要是代码方面有些许差别大家可以选择容易理解的。
BFS:

//岛屿个数
#include <iostream>
#include <cstring>
#include <queue>
#include <string>
using namespace std;//思路:正难则反,我们先不统计岛屿的个数,先将所有的外海标记,那么剩下的没有标记的就是岛屿
//注意,海水是8方走 const int N=53;
int g[N][N]={0};		//0海 1陆 
int book[N][N]={0};		//1表示已经遍历过 
int n,m,k;
int mx[8]={-1,-1,0,1,1,1,0,-1};
int my[8]={0,1,1,1,0,-1,-1,-1};
int imx[4]={0,0,1,-1};
int imy[4]={1,-1,0,0};struct Pos
{int x,y;
};bool isOut(int x,int y)
{if(x<0 || x>=m || y<0 || y>=n)	return true;	//确实出界了if(g[x][y]==1)	return true;	//陆地,不能走return false;	//没有出界 
}bool islandOut(int x,int y)
{if(x<0 || x>=m || y<0 || y>=n)	return true;	//确实出界了//注意这里不需要再判断是否为海洋了,因为我们将这块岛屿内海也视为岛屿 return false;	//没有出界 
}void bfs_island(int x,int y)
{queue<Pos>q;q.push({x,y});book[x][y]=1; while(!q.empty()){Pos cur=q.front();q.pop();for(int i=0;i<4;i++){Pos nex;nex.x=imx[i]+cur.x;nex.y=imy[i]+cur.y;if(islandOut(nex.x,nex.y) || book[nex.x][nex.y]==1)	continue;book[nex.x][nex.y]=1; q.push(nex);}}
}void bfs_sea(int x,int y)
{queue<Pos>q;q.push({x,y});book[x][y]=1;while(!q.empty()){Pos cur=q.front();q.pop();for(int i=0;i<8;i++){Pos nex;nex.x=mx[i]+cur.x;nex.y=my[i]+cur.y;if(isOut(nex.x,nex.y) || book[nex.x][nex.y]==1)	continue;book[nex.x][nex.y]=1; q.push(nex);}}
}int main()
{cin>>k;while(k--){cin>>m>>n;int ans=0;memset(book,0,sizeof(book));memset(g,0,sizeof(g));string str;for(int i=0;i<m;i++){cin>>str;for(int j=0;j<n;j++){if(str[j]=='1')g[i][j]=1;}}for(int i=0;i<n;i++)	//第一行 if(g[0][i]==0 && book[0][i]==0)bfs_sea(0,i);for(int i=0;i<m;i++)	//第一列 if(g[i][0]==0 && book[i][0]==0)bfs_sea(i,0);	for(int i=0;i<n;i++)	//第m-1行 if(g[m-1][i]==0 && book[m-1][i]==0)bfs_sea(m-1,i);for(int i=0;i<m;i++)	//第n-1列 if(g[i][n-1]==0 && book[i][n-1]==0)bfs_sea(i,n-1);for(int i=0;i<m;i++)for(int j=0;j<n;j++){if(book[i][j]==0)	//没标记的,就是岛屿{ans++;bfs_island(i,j); }	}cout<<ans<<endl;}return 0;} 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/17286.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

笔记-python有三种导入模块的方法

其一, import modname &#xff1a; 模块是指一个可以交互使用&#xff0c;或者从另一Python 程序访问的代码段。只要导入了一个模块&#xff0c;就可以引用它的任何公共的函数、类或属性。模块可以通过这种方法来使用其它模块的功能。 用import语句导入模块&#xff0c;就在…

NAT简介

一、NAT 概念定义 NAT&#xff08;Network Address Translation&#xff0c;网络地址转换&#xff09;是一种将私有 IP 地址转换为公有 IP 地址的技术。 允许一个组织内部使用私有 IP 地址的网络通过少量的公有 IP 地址连接到互联网。实现了私有网络与外部网络的通信&#xf…

Angular安装与基础语法

安装Angular cli npm i -g angular-cli 创建项目 npm new project_name(项目名称) 启动项目 cd project_name(j进入新创建的项目) ng serve --open 创建组件 ng g c xxx 会在根组件的统计生成一个组件xxx文件夹&#xff0c;比如ng g c hello 就会生成一个hello组件文件…

夜骑,是探索城市的另一种维度,是与自我对话的静谧时光

月光下的骑行浪漫主义者 想象一下&#xff0c;当白日的喧嚣退去&#xff0c;城市的霓虹开始闪烁&#xff0c;你跨上心爱的自行车&#xff0c;迎着凉爽的晚风&#xff0c;穿梭于灯火阑珊之中。夜骑&#xff0c;不仅仅是一场运动&#xff0c;更是一种逃离日常、探索未知的冒险。但…

【Ambari】Docker 安装Ambari 大数据单机版本

目录 一、前期准备 1.1 部署 docker 1.2 部署 docker-compose 1.3 版本说明 二 、镜像构建启动 2.1 系统镜像构建 2.2 安装包源镜像构建 2.3 kdc镜像构建 2.4 集群安装 2.5 容器导出为镜像 三、Ubuntu环境安装测试 3.1 环境准备 3.2 集群容器启动 一、前期准备 1.…

git多人开发,不用merge的操作方法,阿里codeup

阿里云效 Codeup 是阿里云提供的一款代码托管平台&#xff0c;它集成了代码仓库管理、代码审查、持续集成等功能&#xff0c;非常适合团队协作开发。 通过 Codeup 的合并申请&#xff08;Merge Request&#xff0c;MR&#xff09;功能&#xff0c;可以方便地进行代码合并和审查…

如何在OrangePi AIpro智能小车上实现安全强化学习算法

随着人工智能和智能移动机器人的广泛应用&#xff0c;智能机器人的安全性和高效性问题受到了广泛关注。在实际应用中&#xff0c;智能小车需要在复杂的环境中自主导航和决策&#xff0c;这对算法的安全性和可靠性提出了很高的要求。传统的强化学习算法在处理安全约束时存在一定…

享受当下,还是留待未来?一项fMRI与眼动追踪技术的联合研究

摘要 时间贴现(temporal discount)是指个体对奖励的估计会随着时间流逝而下降的心理现象。具体而言&#xff0c;当获得奖励的时间以日期(日期条件&#xff1b;例如&#xff0c;2023年6月8日)而不是延迟(延迟条件&#xff1b;例如&#xff0c;30天)呈现时&#xff0c;贴现率较低…

免费,Python蓝桥杯等级考试真题--第14级(含答案解析和代码)

Python蓝桥杯等级考试真题–第14级 一、 选择题 答案&#xff1a;B 解析&#xff1a;键为‘B’对应的值为602&#xff0c;故答案为B。 答案&#xff1a;A 解析&#xff1a;字典的符合为花括号&#xff0c;先键后值&#xff0c;故答案为A。 答案&#xff1a;C 解析&#xff1a…

Stable Diffusion 字体和场景结合【把四季藏在海里】

把四季藏在海里&#xff0c;话不多说&#xff0c;上教程。 一、花瓣找字 打开花瓣找下字&#xff0c;也可以自己做&#xff0c;我不会&#xff0c;谢谢。本文用到的字&#xff0c;版权归原作者所有。 白底黑字&#xff1a;那么就是浪花占主要部分&#xff0c;字就是海的效果&…

【改进】YOLOv8 AND YOLOv9 总目录

说明&#xff1a;本专栏为YOLOV8和YOLOV9的使用以及改进的方法。平时比较忙&#xff0c;只能随缘回答问题哈&#xff0c;谨慎订阅&#xff01; &#x1f49b; &#x1f499; &#x1f49c; ❤️ &#x1f49a; &#x1f49b; &#x1f499; &#x1f49c; ❤️ &#x1f49a; &…

sdut java lab 7.1(法二好理解)

7-1 sdut-JAVA-Pig Latin 分数 12 全屏浏览 切换布局 作者 马新娟 单位 山东理工大学 Write a program that requests a word as input, translates the word into Pig Latin and outputs the word input and its equivalent in Pig Latin. The rules for translating a …

C++:STL容器的学习-->string

C:STL容器的学习-->string 1. 构造方法2. string的赋值操作3. string字符串的拼接4. string 查找和替换5. string字符串的比较6. string字符存取7. string 插入和删除8. string截取 需要添加头文件#include <string> 1. 构造方法 string() 创建空的字符串 string(c…

Cordova 环境部署,将 H5 打包为 Android App

Cordova 环境部署&#xff0c;将 H5 打包为 Android App 一、环境配置 1、JDK 环境变量 JAVA_HOME%JAVA_HOME%\bin 2、Android SDK 环境变量 ANDROID_HOMEANDROID_SDK_ROOT%ANDROID_HOME%\build-tools%ANDROID_HOME%\tools%ANDROID_HOME%\platform-tools 3、Node 环境…

C++学习/复习6---内存管理(数据的位置/大小)/new、delete/内存相关面试题(malloc与new/内存泄漏)

一、内存中区域 1.不同数据的存储位置 2.数据占用内存大小 二、动态内存函数 三、new与delete 1.操作内置类型 2.操作自定义类型 四、operator new与operator delete 1.底层源码&#xff08;malloc/free&#xff09; 2.内置/自定义与构造和析构 3.举例 五、定位new表达式 1.举…

使用 Django 和 RabbitMQ 构建高效的消息队列系统

文章目录 RabbitMQ 简介Django 中使用 RabbitMQ总结与拓展 在现代的 Web 应用程序开发中&#xff0c;构建一个高效的消息队列系统变得越来越重要。使用消息队列可以帮助我们解耦系统中不同模块的任务&#xff0c;并提高系统的性能和可扩展性。本文将介绍如何结合 Django 和 Rab…

5.27背单词

formula 公式 conviction 定罪 mischief 恶作剧 counsel 律师 overt 明显的 contagious 有传染性的 dwelling 住所 subtle 不明显的 hazard 危险 vicinity 附近地区 host 大量 dinosaur 恐龙 glamor 魅力 slack 松弛的 gigantic 巨大的 exquisite 精致的 bloody 血的 disastrou…

LeetCode994腐烂的橘子

题目描述 在给定的 m x n 网格 grid 中&#xff0c;每个单元格可以有以下三个值之一&#xff1a;值 0 代表空单元格&#xff1b;值 1 代表新鲜橘子&#xff1b;值 2 代表腐烂的橘子。每分钟&#xff0c;腐烂的橘子 周围 4 个方向上相邻 的新鲜橘子都会腐烂。返回 直到单元格中没…

C++ std::reference_wrapper:让引用更强大

std::reference_wrapper 的通俗易懂解释 一、简介二、std::reference_wrapper 的初衷三、常用示例3.1、与 make_pair 和 make_tuple 一起使用3.2、引用容器3.3、通过 std::thread 按引用传递参数给启动函数3.4、引用作为类成员3.5、按引用传递函数对象3.6、与绑定表达式一起使用…