题目传送门
这道题一看就是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;
}