洛谷P1451题解

本文介绍了如何通过DFS和BFS算法解决一个关于连通块的问题,通过C++代码展示了如何标记二维数组中的连通区域并计算连通块数量。作者首先转换输入的字符串表示的数字,然后分别用深度优先搜索和广度优先搜索来找出每个连通块,最后输出连通块总数。
摘要由CSDN通过智能技术生成

题目传送门

这道题一看就是DFS或BFS。(不会的看这里)

我们看样例就会发现,每行都是一个很大的,可能以零开头的数,

所以我们要用scanf每次只读一位。

但是由于我是蒟蒻,所以我不会,我只会用一种非常XX的方法

string s[110];
void zhuan(){
	for(int i=1;i<=n;i++){
		for(int j=0;j<m;j++){
			a[i][j+1]=s[i][j]-'0';
		}
	}
}
for(int i=1;i<=n;i++){
		cin>>s[i];
	}zhuan();

 思路:

这道题是求连通块的一道题。

什么是连通块呢?

连通块就是由一系列相同特征的方块组成的连续区域。

比如,本题中的“相同特征的方块”就是1~9。

那该怎么求呢?

我们可以用DFS或BFS来标记一个连通块内的所有数字,再用for循环遍历一遍该数组,找到连通块上的一个点,再调用DFS或BFS,将这个连通块的所有点都标记一遍,同时ans++

完整代码:

DFS:

#include<iostream>
#include<string> 
using namespace std;
int n,m,a[110][110],vis[110][110],cnt;
string s[110];
int nx[5]={-1,1,0,0};
int ny[5]={0,0,-1,1};
void zhuan(){
	for(int i=1;i<=n;i++){
		for(int j=0;j<m;j++){
			a[i][j+1]=s[i][j]-'0';
		}
	}
}void dfs(int x,int y){
	vis[x][y]=1;
	for(int i=0;i<=3;i++){
		int xx=x+nx[i];
		int yy=y+ny[i];
		if(xx>=1&&yy>=1&&xx<=n&&yy<=m&&!vis[xx][yy]&&a[xx][yy]!=0){
			vis[xx][yy]=1;
			dfs(xx,yy);
		}
	}
}
int main(){
	cin>>n>>m;
	for(int i=1;i<=n;i++){
		cin>>s[i];
	}zhuan();
	for(int i=1;i<=n;i++){
		for(int j=1;j<=m;j++){
			if(a[i][j]&&!vis[i][j]){
				cnt++;
				dfs(i,j);
			}
		}
	}
	cout<<cnt;
	return 0;
}

BFS:

#include<iostream>
#include<string> 
#include<queue>
#include<map>
using namespace std;
int n,m,a[110][110],vis[110][110],cnt;
string s[110];
int nx[5]={-1,1,0,0};
int ny[5]={0,0,-1,1};
queue<pair<int,int> > q;
void zhuan(){
	for(int i=1;i<=n;i++){
		for(int j=0;j<m;j++){
			a[i][j+1]=s[i][j]-'0';
		}
	}
}void bfs(int x,int y){
	q.push(make_pair(x,y));
	vis[x][y]=1;
	while(!q.empty()){
		for(int i=0;i<=3;i++){
		int xx=q.front().first+nx[i];
		int yy=q.front().second+ny[i];
		if(xx>=1&&yy>=1&&xx<=n&&yy<=m&&!vis[xx][yy]&&a[xx][yy]!=0){
			vis[xx][yy]=1;
			q.push(make_pair(xx,yy));
		}
	}q.pop();
	}
	
}
int main(){
	cin>>n>>m;
	for(int i=1;i<=n;i++){
		cin>>s[i];
	}zhuan();
	for(int i=1;i<=n;i++){
		for(int j=1;j<=m;j++){
			if(a[i][j]&&!vis[i][j]){
				cnt++;
				bfs(i,j);
			}
		}
	}cout<<cnt;
	return 0;
}

AC记录 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值