岛屿个数——代码

这篇文章介绍了如何使用C++编程语言实现海域搜索算法,通过BFS方法遍历岛屿并扩展海水区域,确保不会遗漏任何岛屿,解决岛屿是否能形成环的问题。
摘要由CSDN通过智能技术生成
// 只要海水搜8个方向 可以搜到的岛屿 都不在环内(如果在环内 海水是进不去的)
// 如果搜到岛屿 岛屿个数加一 再把它连接的岛屿都搜到 防止多次重复计算
//观察样例二 最外圈岛屿无法成环是因为它有一个缺口 使海水可以进去 进而又形成了一个新的岛屿
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef pair<ll,ll>PII;
const ll N=110;
ll dxx[8]={-1,-1,0,1,1,1,0,-1};
ll dyy[8]={0,1,1,1,0,-1,-1,-1};
ll dx[4]={-1,0,1,0};
ll dy[4]={0,-1,0,1};
ll n,m,cnt=0;
char Map[N][N];
bool st[N][N];
void bfs1(ll x,ll y)//搜岛屿bfs板子
{
    cnt++;
    queue<PII>q;
    st[x][y]=1;
    q.push({x,y});
    while(q.size())
    {
        auto t=q.front();
        q.pop();
        for(ll i=0;i<4;i++)
        {
            ll tx=t.first+dx[i];
            ll ty=t.second+dy[i];
            if(tx>=0&&tx<=n+1&&ty>=0&&ty<=m+1&&Map[tx][ty]=='1'&&!st[tx][ty])
            {
                st[tx][ty]=1;
                q.push({tx,ty});
            }
        }
    }
}
void bfs2(ll x1,ll y1)//正常搜海水bfs板子
{
    queue<PII>q1;
    q1.push({x1,y1});
    st[x1][y1]=1;
    while(q1.size())
    {
        auto t1=q1.front();
        q1.pop();
        for(ll i=0;i<8;i++)
        {
            ll tx1=t1.first+dxx[i];
            ll ty1=t1.second+dyy[i];
            if(tx1>=0&&tx1<=n+1&&ty1>=0&&ty1<=m+1&&!st[tx1][ty1])
            {
                if(Map[tx1][ty1]=='1')bfs1(tx1,ty1);
                else 
                {
                    st[tx1][ty1]=1;
                    q1.push({tx1,ty1});
                }
            }
        }
    }
}
void solve(ll n,ll m)
{
    for(ll i=1;i<=n;i++)
    {
        for(ll j=1;j<=m;j++)
        {
            cin>>Map[i][j];
        }
    }
    bfs2(0,0);//从外圈扩展海水搜索 避免卡图
}
int main()
{
    ll T;
    cin>>T;
    while(T--)
    {
        cin>>n>>m;
        cnt=0;
        memset(st,0,sizeof(st));
        memset(Map,0,sizeof(Map));
        //在原地图外阔一圈海水
        //防止角落 岛屿卡住搜索的范围
        solve(n,m);
        cout<<cnt<<endl;
    }
    return 0;
}

  • 4
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值