10 bfs(flood fill与最短路模型)

简介: 10 bfs(flood fill与最短路模型)

1 flood fill

1.Lake Counting

信息学奥赛一本通(C++版)在线评测系统 (ssoier.cn)

#include<bits/stdc++.h>
using namespace std;
#define x first
#define y second
typedef pair<int,int> pii;
const int N=115;
int n,m,cnt=0;
char g[N][N];
bool st[N][N];
pii q[N*N];
void bfs(int sx,int sy)
{
    int hh=0,tt=0;
    q[0]={sx,sy};
    st[sx][sy]=true;
    while(hh<=tt)
    {
        auto t=q[hh++];
        for(int i=t.x-1;i<=t.x+1;i++)//因为求八个周围的,则循环枚举两层即可
            for(int j=t.y-1;j<=t.y+1;j++)
            {
               if(i<0||i>=n||j<0||j>=m) continue;//假如越界则continue
               if(st[i][j]) continue;//假如标记过则continue
               if(g[i][j]=='.') continue;//假如是干燥的则continue
               st[i][j]=true;//标记这个点被标记过
               q[++tt]={i,j};//把这个积水放进队列用,以后用他来更新其他的积水
            }
    }
}
int main()
{
  cin>>n>>m;
  for(int i=0;i<n;i++) cin>>g[i];
  for(int i=0;i<n;i++)//遍历每一个点
    for(int j=0;j<m;j++)
      if(!st[i][j]&&g[i][j]=='W')//假如之前没标记过并且是新的水洼
      {
          bfs(i,j);//遍历这个水洼覆盖的水洼
          cnt++;//水洼数量加一
      }
  cout<<cnt<<endl;
   return 0;
}

2.The Castle

信息学奥赛一本通(C++版)在线评测系统 (ssoier.cn)

#include<bits/stdc++.h>
using namespace std;
#define x first
#define y second
typedef pair<int,int> pii;
const int N=55;
int n,m,cnt=0,ans=0;
int g[N][N];
bool st[N][N];
pii q[N*N];
int dx[4]={0,-1,0,1},dy[4]={-1,0,1,0};//按照题目的方向西北东南的方向
int bfs(int sx,int sy)
{
    int hh=0,tt=0;
    q[0]={sx,sy};
    st[sx][sy]=true;
    while(hh<=tt)
    {
        auto t=q[hh++];//取出队头
       for(int i=0;i<4;i++)//枚举西北东南四个方向
            {
                int a=t.x+dx[i],b=t.y+dy[i];//下一步
               if(a<0||a>=n||b<0||b>=m) continue;//假如越界则continue
               if(st[a][b]) continue;//假如标记过则continue
               if((g[t.x][t.y]>>i)&1) continue;//假如是墙的话则continue
               st[a][b]=true;//标记这个点被标记过
               q[++tt]={a,b};//把这个点放进队列用,以后用他来更新其他的房间
            }
    }
   return tt+1;//返回房间数量
}
int main()
{
  cin>>n>>m;
  for(int i=0;i<n;i++)
    for(int j=0;j<m;j++)
       cin>>g[i][j];
  for(int i=0;i<n;i++)//遍历每一个点
    for(int j=0;j<m;j++)
      if(!st[i][j])//假如之前没标记过,则是新的房间
      {
          ans=max(ans,bfs(i,j));//更新一遍房间的最大值
          cnt++;//水洼数量加一
      }
  cout<<cnt<<endl;
  cout<<ans<<endl;
   return 0;
}

3.山峰和山谷

信息学奥赛一本通(C++版)在线评测系统 (ssoier.cn)

#include<bits/stdc++.h>
using namespace std;
#define x first
#define y second
typedef pair<int,int> pii;
const int N=1e3+10;
int n;
int g[N][N];
bool st[N][N];
pii q[N*N];
int bfs(int sx,int sy)
{
    int hh=0,tt=0;
    bool up=false,low=false;//用来记录是不是山峰或者山谷
    st[sx][sy]=true;
    q[0]={sx,sy};
    while(hh<=tt)
    {
        auto t=q[hh++];
        //因为是要周围八个方向,所以用枚举的方法
        for(int i=t.x-1;i<=t.x+1;i++)
            for(int j=t.y-1;j<=t.y+1;j++)
        {
            if(i<0||i>=n||j<0||j>=n) continue;//假如越界
            if(g[i][j]>g[t.x][t.y]) low=true;//假如是山谷
            else if(g[i][j]<g[t.x][t.y]) up=true;//假如是山峰
            else if(!st[i][j])//假如是同一高度且没被标记过的
            {
                st[i][j]=true;
                q[++tt]={i,j};
            }
        }
    }
    if(up&&low) return -1;//假如即使山峰并且又是山谷,则啥也不是
    else if(up) return 1;//假如山峰
    else if(low) return 0;//假如山谷
    else return 2;//假如二者都可以
}
int main()
{
    int up=0,low=0;
   cin>>n;
   for(int i=0;i<n;i++)
    for(int j=0;j<n;j++)
       cin>>g[i][j];
   for(int i=0;i<n;i++)
    for(int j=0;j<n;j++)
      if(!st[i][j])//假如这个点没有被走过
      {
          int t=bfs(i,j);
         if(t==1) up++;//假如是山峰
         else if(t==0) low++;//假如是山谷
         else if(t==2) up++,low++;//假如即可以作为山峰,又可以作为山谷
      }
    cout<<up<<" "<<low<<endl;
   return 0;
}

2最短路模型

1.迷宫问题

信息学奥赛一本通(C++版)在线评测系统 (ssoier.cn)

#include<bits/stdc++.h>
using namespace std;
#define x first
#define y second
typedef pair<int,int> pii;
const int N=10;
int n;
int g[N][N];
bool st[N][N];//标记那个点被走过了
pii q[N*N],pre[N][N];//pre用来记录某个点的上一步
int dx[4]={-1,0,1,0},dy[4]={0,1,0,-1};
void bfs(int sx,int sy)
{
    int hh=0,tt=0;
    st[sx][sy]=true;
    q[0]={sx,sy};
    pre[sx][sy]={0,0};
    while(hh<=tt)
    {
        auto t=q[hh++];
       for(int i=0;i<4;i++)//枚举四个方向
       {
           int a=t.x+dx[i],b=t.y+dy[i];
           if(a<0||a>=n||b<0||b>=n) continue;//假如越界
           if(g[a][b]) continue;//假如是墙,不能走
           if(st[a][b]) continue;//假如之前走过这个点
           q[++tt]={a,b};
           st[a][b]=true;
           pre[a][b]=t;//标记这个点由t转移过来的
        }
    }
}
int main()
{
    n=5;
   for(int i=0;i<n;i++)
    for(int j=0;j<n;j++)
         cin>>g[i][j];
   bfs(n-1,n-1);//倒序求最短路
   pii end(0,0);//正向输出路径
   while(1)
   {
       printf("(%d, %d)\n",end.x,end.y);
       if(end.x==n-1&&end.y==n-1) break;
       end=pre[end.x][end.y];
   }
   return 0;
}

2.武士风度的牛

188. 武士风度的牛 - AcWing题库

#include<bits/stdc++.h>
using namespace std;
#define x first
#define y second
typedef pair<int,int> pii;
const int N=160;
int n,m;
char g[N][N];
int dist[N][N];//记录吗,每个点到起点的步数
pii q[N*N];
int dx[]={-1,1,2,2,1,-1,-2,-2},dy[]={2,2,1,-1,-2,-2,-1,1};//题目中的八个方向
int bfs(int sx,int sy,int ex,int ey)
{
    int hh=0,tt=0;
    q[0]={sx,sy};
    memset(dist,-1,sizeof dist);
    dist[sx][sy]=0;
    while(hh<=tt)
    {
        auto t=q[hh++];
       for(int i=0;i<8;i++)//枚举八个方向
       {
           int a=t.x+dx[i],b=t.y+dy[i];
           if(a<0||a>=n||b<0||b>=m) continue;//假如越界
           if(g[a][b]=='*') continue;//假如是障碍,不能走
           if(dist[a][b]!=-1) continue;//假如之前已经更新过这个点
           q[++tt]={a,b};
           dist[a][b]=dist[t.x][t.y]+1;//则步数+1
        }
    }
    return dist[ex][ey];//返回到终点的步数
}
int main()
{
   cin>>m>>n;
   int sx,sy,ex,ey;
   for(int i=0;i<n;i++)
    for(int j=0;j<m;j++)
   {
       cin>>g[i][j];
       if(g[i][j]=='K') sx=i,sy=j;//标记起点
       if(g[i][j]=='H') ex=i,ey=j;//标记终点
   }
   cout<<bfs(sx,sy,ex,ey)<<endl;//输出最小步数
   return 0;
}

3.抓住那头牛

信息学奥赛一本通(C++版)在线评测系统 (ssoier.cn)

#include<bits/stdc++.h>
using namespace std;
const int N=2e5+10;
int n,k;
int dist[N];//记录每个点到起点的步数
int q[N];
int bfs()
{
    int hh=0,tt=0;
    q[0]=n;
    memset(dist,-1,sizeof dist);
    dist[n]=0;
    while(hh<=tt)
    {
        int t=q[hh++];
        if(t==k) return dist[k];//返回到起点的步数
        if(t+1<N&&dist[t+1]==-1)//第一种情况走,且该位置没有被走过
        {
            dist[t+1]=dist[t]+1;//则该点距离+1
            q[++tt]=t+1;//将该点放进队列中
        }
        if(t-1>=0&&dist[t-1]==-1)//第二种情况走,且该位置没有被走过
        {
            dist[t-1]=dist[t]+1;//则该点距离+1
            q[++tt]=t-1;//将该点放进队列中
        }
        if(2*t<N&&dist[2*t]==-1)//第三种情况走,且该位置没有被走过
        {
            dist[2*t]=dist[t]+1;//则该点距离+1
            q[++tt]=2*t;//将该点放进队列中
        }
    }
    return 0;
}
int main()
{
   cin>>n>>k;
   cout<<bfs()<<endl;//输出最小步数
   return 0;
}
相关文章
|
19天前
|
算法
Floyd 最短路径【学习算法】
Floyd 最短路径【学习算法】
33 0
|
19天前
E. Nearest Opposite Parity(多源最短路bfs)
E. Nearest Opposite Parity(多源最短路bfs)
|
19天前
|
算法 定位技术 索引
class064 Dijkstra算法、分层图最短路【算法】
class064 Dijkstra算法、分层图最短路【算法】
33 0
|
7月前
|
算法
算法学习--最短路问题
算法学习--最短路问题
|
8月前
|
机器学习/深度学习 算法 测试技术
C++算法:01BFS最短距离的原理和实现
C++算法:01BFS最短距离的原理和实现
|
机器学习/深度学习 定位技术
【每日一题Day109】LC1210穿过迷宫的最少移动次数 | BFS+dp
思路:使用BFS搜索,队列中存放三元组[蛇尾的横轴坐标x,蛇尾的纵轴坐标y,蛇的状态],当蛇为水平时,状态为0;当蛇为竖直时,状态为1
98 1
【每日一题Day109】LC1210穿过迷宫的最少移动次数 | BFS+dp
CF1272 E.Nearest Opposite Parity(反向建图+BFS)
CF1272 E.Nearest Opposite Parity(反向建图+BFS)
78 0
CF1272 E.Nearest Opposite Parity(反向建图+BFS)
UPC Graph (最小生成树 || 并查集+二分)
UPC Graph (最小生成树 || 并查集+二分)
73 0
Floyd-Warshall算法 五行解决 (最短路径)
Floyd-Warshall算法 五行解决 (最短路径)
Floyd-Warshall算法 五行解决 (最短路径)
|
机器学习/深度学习 算法
【刷穿 LeetCode】求「连通图经过所有点的最短路径」的三种方式 :「BFS」&「Floyd + 状压 DP」 &「AStar 算法」
【刷穿 LeetCode】求「连通图经过所有点的最短路径」的三种方式 :「BFS」&「Floyd + 状压 DP」 &「AStar 算法」