【蓝桥杯每日一题】4.4 扫雷

news/2024/7/20 21:49:58 标签: 蓝桥杯, 算法, 图论, 深度优先, c语言, c++

题目来源:

687. 扫雷 - AcWing题库
参考:y总视频讲解

问题描述:

  • 找到解决扫雷游戏中的最小点击次数

思考:

为了保证胜利且每个格子只能走一次,所以当遇到地雷或检测到该格子周围存在地雷的时候就需要停止搜索,那么如何知道当前的格子的周围是否存在地雷呢?

可以用一个二维数组 n u m num num来统计每个位置周围的地雷数量,例如,当前位置为 ( 2 , 2 ) (2,2) (2,2)则就需要查看 ( 1 , 1 ) (1,1) (1,1) ( 1 , 2 ) (1,2) (1,2) ( 1 , 3 ) (1,3) (1,3) ( 2 , 1 ) (2,1) (2,1) ( 2 , 3 ) (2,3) (2,3) ( 3 , 1 ) (3,1) (3,1) ( 3 , 2 ) (3,2) (3,2) ( 3 , 3 ) (3,3) (3,3)的位置是否存在地雷,如果存在就把 ( 2 , 2 ) (2,2) (2,2)位置的地雷数量 + 1 +1 +1。由于题目中求最小点击次数,在我们每次点击 0 0 0位置时,其周围的所有 0 0 0都会被点亮,所以我们可以把每个为 0 0 0的位置进行搜索并进行标记表示已经扫描过了一次,并且将点击次数 + 1 +1 +1。最后再进行一次遍历,将所有状态不为 0 0 0的格子进行点击即可求出最小点击次数。

步骤:

  • 使用num数组来表示当前位置周围的地雷数量,如果当前位置就是地雷,那么设置为-1。
  • 遍历num数组,如果当前位置为0,即周围没有地雷,则进行搜索,将其周围所有为0的点进行标记,直到num不为0时停止。
    最后统计没有被标记为-1的结点数量,累加,即为最后结果

AC Code:

#include<bits/stdc++.h>
using namespace std;

const int N=310;
int dx[8]={1,1,1,0,0,-1,-1,-1};
int dy[8]={1,0,-1,1,-1,1,0,-1};
char m[N][N];  //整个的地图
int num[N][N];  //显示的数字
bool vis[N][N]; //遍历的记录

int t,n;
int res=0;

void dfs(int x,int y)  //遍历0周围所有的点,标记并且将该点设置为不可点击
{
	vis[x][y]=1;
	num[x][y]=-1;
	
	for(int i=0;i<8;i++)
	{
		int nx=x+dx[i],ny=y+dy[i];
		if(nx>=0&&ny>=0&&nx<n&&ny<n) //边界值的判断
		{
			if(!vis[nx][ny]&&num[nx][ny]!=-1)
			{
				if(num[nx][ny]>0) //标记已经遍历
				{
					vis[nx][ny]=1;
					num[nx][ny]=-1;
				}
				if(num[nx][ny]==0) dfs(nx,ny);
			}
		}
	}
}

int main()
{
	scanf("%d",&t);
	for(int i=1;i<=t;i++)
	{
		
		scanf("%d",&n);
		for(int j=0;j<n;j++) cin>>m[j];
		
		memset(num,0,sizeof(num)); //每次循环记得清空数组
		
		for(int j=0;j<n;j++)
		{
			for(int k=0;k<n;k++)
			{
				if(m[j][k]!='*') //非雷点统计其8邻域中雷的个数
				{
					for(int d=0;d<8;d++)
					{
						int x=j+dx[d],y=k+dy[d];
						if(x>=0 && x<n && y>=0 && y<n && m[x][y]=='*') //找雷
						{
							num[j][k]++;
						}
					}
				}else{
					num[j][k]=-1; //是雷则标记为-1
				}
				
			}
		}
		
		res=0; //重置res
		memset(vis,0,sizeof(vis)); //重置遍历数组
		
		//找到0的个数
		for(int j=0;j<n;j++){
			for(int k=0;k<n;k++){
				if(num[j][k]==0&&!vis[j][k]){ 
					res++;
					dfs(j,k);
				}
			}
		}
		
		//找漏网之鱼
		for(int j=0;j<n;j++)
		{
			for(int k=0;k<n;k++){
				if(num[j][k]!=-1&&!vis[j][k]) res++;
			}
		}
		printf("Case #%d: %d\n",i,res);
	}
	
	
	return 0;
}

明日预告:网络分析
准备趁着清明节假期啃块硬骨头。


http://www.niftyadmin.cn/n/5471034.html

相关文章

JAVAEE——文件IO

文章目录 文件的概念什么是文件&#xff1f;树型结构组织 和 目录文件路径相对路径绝对路径 文件的分类文件的权限 文件读写IO API字符流操作API 警告字节流操作APIInputStreamOutputStream 文件的概念 什么是文件&#xff1f; 我们先来理解一下什么是文件&#xff0c;那么想…

Java中常用的加密算法及其实现原理详解(二)

本系列文章简介&#xff1a; 随着互联网的快速发展&#xff0c;信息的安全保护愈发重要。在软件开发中&#xff0c;加密算法被广泛应用于数据的加密和解密过程中&#xff0c;以保护敏感信息的机密性和完整性。Java作为一种广泛应用于企业级开发的编程语言&#xff0c;也提供了丰…

Spring重点知识(个人整理笔记)

目录 1. 为什么要使用 spring&#xff1f; 2. 解释一下什么是 Aop&#xff1f; 3. AOP有哪些实现方式&#xff1f; 4. Spring AOP的实现原理 5. JDK动态代理和CGLIB动态代理的区别&#xff1f; 6. 解释一下什么是 ioc&#xff1f; 7. spring 有哪些主要模块&#xff1f;…

为什么android创建Fragment推荐用newInstance

FullScreenDialogFragment使用newInstance方法不是因为它是一个单例&#xff0c;而是因为这是创建DialogFragment实例并同时提供参数的一种标准模式。这种模式通常称为静态工厂方法模式&#xff0c;在Android开发中被广泛使用&#xff0c;尤其是用于Fragment的实例化。 newIns…

基于ACO蚁群优化的VRPSD问题求解matlab仿真,输出规划路径结果和满载率

目录 1.程序功能描述 2.测试软件版本以及运行结果展示 3.核心程序 4.本算法原理 4.1 关键概念与模型 4.2数学模型 5.完整程序 1.程序功能描述 基于ACO蚁群优化的VRPSD问题求解matlab仿真,输出ACO优化的收敛曲线&#xff0c;规划路径结果和每一条路径的满载率。 2.测试软…

互联网、电商行业实时大数据分析最佳实践|实时大数据分析最重要的数据采集|电商API采集接口

电商API采集接口的分类 此API目前支持以下基本接口&#xff1a; item_get 获得淘宝商品详情item_get_pro 获得淘宝商品详情高级版item_review 获得淘宝商品评论item_fee 获得淘宝商品快递费用item_password 获得淘口令真实urlitem_list_updown 批量获得淘宝商品上下架时间selle…

微电网优化:基于光学显微镜算法(Optical microscope algorithm,OMA)的微电网优化(提供MATLAB代码)

一、微电网优化模型 微电网是一个相对独立的本地化电力单元&#xff0c;用户现场的分布式发电可以支持用电需求。为此&#xff0c;您的微电网将接入、监控、预测和控制您本地的分布式能源系统&#xff0c;同时强化供电系统的弹性&#xff0c;保障您的用电更经济。您可以在连接…

【C++】每日一题 169 多数元素

给定一个大小为 n 的数组 nums &#xff0c;返回其中的多数元素。多数元素是指在数组中出现次数 大于 ⌊ n/2 ⌋ 的元素。 你可以假设数组是非空的&#xff0c;并且给定的数组总是存在多数元素。 #include <vector>class Solution { public:int majorityElement(std::v…