广度优先搜索C++算法习题练习

本文通过多个实际问题展示了广度优先搜索(BFS)算法的应用,包括寻找棋盘上两种走法最少步数、计算城堡房间数量及最大房间、数池塘、走迷宫以及走出迷宫等。每个问题都给出了详细的输入输出示例,并提供了相应的C++代码实现。这些实例有助于理解BFS在解决实际问题中的核心思想和步骤。
摘要由CSDN通过智能技术生成

因为本文章是习题练习,所有不会有具体讲解。如想看具体的思路讲解,请在我的博客中找到广度优先搜索算法带图详解, 或者访问

广度优先搜索算法带图详解_不怕困难的博客的博客-CSDN博客

问题 A: 【一本通基础广度优先搜索】 最少步数

[题目描述]

在各种棋中,一种棋子的走法总是一定的,如中国象棋中马走“日”。有一位小学生就想如果马能有两种走法将更加增加趣味性,因此,他规定马既能按“日”飞,也能各象一样走“田”字。他的同桌平时喜欢下围棋,知道这件事后觉得很有趣,就和他玩一种新游戏,在围棋盘上任选两点A、B,A点放上黑子,B点放上白子,代表两匹马。棋子可以按“日”字走,也可以按“田”字走,俩人一个走黑马,一个走白马。谁用最少的步数走到左上角坐标为(1,1)的点时,谁获胜。现在他请你帮忙,给你A,B两点的坐标,想知道两个位置到(1,1)点的可能最少步数。

输入

每行的两个整数,第一行为A点坐标

第二行为B点坐标

(* 注意:棋盘大小规模为100*100以内 *)

输出

两行第一行为A点到(1,1)的步数

第二行为B点到(1,1)的步数

样例输入

12 16
18 10

样例输出

8
9

代码:

#include <bits/stdc++.h>
using namespace std;
 
int dx[12] = {-2, -2, -1, 1, 2, 2, 2, 2, 1, -1, -2, -2};
int dy[12] = {-1, -2, -2, -2, -2, -1, 1, 2, 2, 2, 2, 1};
 
int main()
{
    int s[101][101], que[10000][4] = {0}, x1, x2, y1, y2;
    memset(s, 0xff, sizeof((s)));//s数组的初始化 
    int head = 1, tail = 1;//初始位置入队列 
    que[1][1] = 1;
    que[1][2] = 1;
    que[1][3] = 0;
    cin >> x1 >> y1 >> x2 >> y2;//读入黑马和白马的位置 
    while(head <= tail)//如果队列为空, 则拓展队首结点 
    {
        for(int i = 0; i <= 11; i++)//枚举12个拓展方向 
        {
            int x = que[head][1] + dx[i];
            int y = que[head][2] + dy[i];
            if(x > 0 && y > 0 && x <= 100 && y <= 100)
            {
                if(s[x][y] == -1)
                {
                    s[x][y] = que[head][3] + 1;
                    tail++;//(1, 1)到(x,y)的最小步数入队 
                    que[tail][1] = x;
                    que[tail][2] = y;
                    que[tail][3] = s[x][y];
                    if(s[x1][y1] > 0 && s[x2][y2] > 0)//输出答案 
                    {
                        cout << s[x1][y1] << endl;
                        cout << s[x2][y2] << endl;
                        return 0;
                    }
                }
            }
        }
        head++;
    }
    return 0;
} 

问题 B: 【一本通基础广度优先搜索】The Castle

【题目描述】

一座城堡被分成m*n个方块(m≤50,n≤50),每个方块可有0~4堵墙(0表示无墙)。下面示出了建筑平面图:

图中的加粗黑线代表墙。几个连通的方块组成房间,房间与房间之间一定是用黑线(墙)隔开的。

现在要求你编一个程序,解决以下2个问题:

1、该城堡中有多少个房间?

2、最大的房间有多大?

【输入】

平面图用一个数字表示一个方块(第1个房间用二进制1011表示,0表示无东墙,用十进制11表示)。

第一行一个整数m(m≤50),表示房子南北方向的长度。

第二行一个整数n(n≤50),表示房子东西方向的长度。

后面的m行,每行有n个整数,每个整数都表示平面图对应位置的方块的特征。每个方块中墙的特征由数字P来描述(0≤P≤15)。数字P是下面的可能取的数字之和:

1(西墙 west)

2(北墙 north)

4(东墙 east)

8(南墙 south)

室内的墙被定义两次: 例如方块(1,1)中的南墙也被位于其南面的方块(2,1)定义了一次。

建筑中至少有两个房间。

【输出】

第1行:一个整数,表示房间总数;

第2行:一个整数,表示最大房间的面积(方块数)。

【输入样例】

4
7
11 6 11  6  3 10  6
7  9  6 13  5 15  5
1 10 12  7 13  7  5
13 11 10 8 10 12 13

【输出样例】

5
9

#include <bits/stdc++.h>
using namespace std;
 
int dx[4] = {-1, 1, 0, 0};
int dy[4] = {0, 0, -1, 1};
int a[51][51][4], q[5001][3];
int tot, i, j, p, n, m, maxx = -1;
bool b[51][51];
 
void bfs(int x, int y)
{
    memset(q, 0, sizeof(q));
    tot++;
    int head = 0, tail = 1, xx, yy;
    q[1][1] = x;
    q[1][2] = y;
    b[x][y] = false;
    int cnt = 1;
    while(head < tail)
    {
        head++;
        for(int i = 0; i <= 3; i++)
        {
            xx = q[head][1] + dx[i];
            yy = q[head][2] + dy[i];
            if((xx > 0) && (xx <= m) && (yy > 0) && (yy <= n) && (b[xx][yy]) && a[q[head][1]][q[head][2]][i] == 0)
            {
                tail++;
                cnt++;
                b[xx][yy] = false;
                q[tail][1] = xx;
                q[tail][2] = yy;
            }
        }
    }
    maxx = max(cnt, maxx);
}
 
int main()
{
    memset(b, true, sizeof(b));
    scanf("%d %d", &m, &n);
    for(int i = 1; i <= m; i++)
    {
        for(int j = 1; j <= n; j++)
        {
            scanf("%d", &p);
            if(p >= 8)
            {
                p -= 8;
                a[i][j][1] = 1;
            }
            if(p >= 4)
            {
                p -= 4;
                a[i][j][3] = 1;
            }
            if(p >= 2)
            {
                p -= 2;
                a[i][j][0] = 1;
            }
            if(p >= 1)
            {
                p -= 1;
                a[i][j][2] = 1;
            }
        }
    }
    for(int i = 1; i <= m; i++)
    {
        for(int j = 1; j <= n; j++)
        {
            if(b[i][j])
            {
                bfs(i, j);
            }
        }
    }
    printf("%d\n%d\n", tot, maxx);
    return 0;
}

问题 C: 【一本通基础广度优先搜索】Lake Counting 数池塘

[题目描述]

 农夫约翰的农场可以表示成N×M(1≤N,M≤100)个方格组成的矩形.由于近日的降雨,

在约翰农场上的不同地方形成了池塘.每一个方格或者有积水(’W’)或者没有积水(’.’).农夫约翰打算数出他的农场上共形成了多少池塘.一个池塘是一系列相连的有积水的方格,每一个方格周围的八个方格都被认为是与这个方格相连的.

    现给出约翰农场的图样,要求输出农场上的池塘数.

输入

 第1行:由空格隔开的两个整数N和M.

 第2到N+1行:每行M个字符代表约翰农场的一排方格的状态.每个字符或者是’W’或者

是’.’,字符之间没有空格.

输出

约翰农场上的池塘数.

样例输入

10 12
W ........ WW.
. WWW ..... WWW
.... WW ... WW.
......... WW.
......... W..
..W ...... W..
.W.W ..... WW.
W.W.W ..... W.
.W.W ...... W.
..W ....... W.

样例输出

3
#include <bits/stdc++.h>
using namespace std;
 
int dx[8] = {-1, -1, 0, 1, 1, 1, 0, -1};
int dy[8] = {0, 1, 1, 1, 0, -1, -1, -1};
int n, m;
int ans = 0;
int h[10001][3];
char map1[101][101];
 
void js(int p, int q)
{
    memset(h, 0, sizeof(h));
    int x, y, head = 0, tail = 1;
    ans++;
    h[1][1] = p;
    h[1][2] = q;
    map1[p][q] = '.';
    do
    {
        head++;
        for(int i = 0; i < 8; i++)
        {
            x = h[head][1] + dx[i];
            y = h[head][2] + dy[i];
            if(x > 0 && x <= n && y > 0 && y <= m && map1[x][y] == 'W')
            {
                map1[x][y] = '.';
                tail++;
                h[tail][1] = x;
                h[tail][2] = y;
            }
        }
    }while(head < tail);
}
 
int main()
{
    cin >> n >> m;
    for(int i = 1; i <= n; i++)
    {
        for(int j = 1; j <= m; j++)
        {
            cin >> map1[i][j];
        }
    }
    for(int i = 1; i <= n; i++)
    {
        for(int j = 1; j <= m; j++)
        {
            if(map1[i][j] == 'W')
            {
                js(i, j);
            }
        }
    }
    cout << ans << endl;
    return 0;
}

问题 D: 【一本通基础广度优先搜索】走迷宫

[题目描述]

一个迷宫由R行C列格子组成,有的格子里有障碍物,不能走;有的格子是空地,可以走。

给定一个迷宫,求从左上角走到右下角最少需要走多少步(数据保证一定能走到)。只能在水平方向或垂直方向走,不能斜着走。

输入

第一行是两个整数,R和C,代表迷宫的长和宽。( 1≤ R,C ≤ 40)

接下来是R行,每行C个字符,代表整个迷宫。

空地格子用‘.’表示,有障碍物的格子用‘#’表示。

迷宫左上角和右下角都是‘.’。

输出

输出从左上角走到右下角至少要经过多少步(即至少要经过多少个空地格子)。计算步数要包括起点和终点。

样例输入

5 5
..###
#....
#.#.#
#.#.#
#.#..

样例输出

9
#include <bits/stdc++.h>
using namespace std;
int xx[4] = {-1, 1, 0, 0};
int yy[4] = {0, 0, -1, 1};
int h[1001][5];
bool a[41][41];
int r, c, x, y;
char k;
int head, tail, i;
int main()
{
    cin >> r >> c;
    for(int i = 1; i <= r; i++)
    {
        for(int j = 1; j <= c; j++)
        {
            cin >> k;
            if(k == '.') a[i][j] = true;
            else a[i][j] = false;
        }
    }
//  for(int i = 1; i <= r; i++)
//  {
//      for(int j = 1; j <= c; j++)
//      cout << a[i][j] << " ";
//      cout << endl;
//  }
    head = 0;
    tail = 1;
    h[1][1] = 1;
    h[1][2] = 1;
    h[1][3] = 1;
    while(head < tail)
    {
        head++;
        for(int i = 0; i < 4; i++)
        {   
            x = h[head][1] + xx[i];
            y = h[head][2] + yy[i];
            if((x > 0) && (x <= r) && (y > 0) && (y <= c) && (a[x][y] == true))
            {
                tail++;
                h[tail][1] = x;
                h[tail][2] = y;
                h[tail][3] = h[head][3] + 1;
                a[x][y] = false;
                if(x == r && y == c)
                {
                    cout << h[tail][3] << endl;
                    return 0;
                }
            }
        }
         
    }
    return 0;
}

问题 E: 【一本通基础广度优先搜索】走出迷宫

[题目描述]

当你站在一个迷宫里的时候,往往会被错综复杂的道路弄得失去方向感,如果你能得到迷宫地图,事情就会变得非常简单。

假设你已经得到了一个n*m的迷宫的图纸,请你找出从起点到出口的最短路。

输入

第一行是两个整数n和m(1≤n,m≤100),表示迷宫的行数和列数。

接下来n行,每行一个长为m的字符串,表示整个迷宫的布局。字符‘.’表示空地,‘#’表示墙,‘S’表示起点,‘T’表示出口。

输出

输出从起点到出口最少需要走的步数。

样例输入

3 3
S#T
.#.
...

样例输出

6
#include <bits/stdc++.h>
using namespace std;
int h[1004][4];
int xx[4] = {-1, 1, 0, 0},
    yy[4] = {0, 0, -1, 1};
bool a[101][101];
int main()
{
    int n, m, fx, fy, x, y, t, w;
    char c;
    memset(a, true, sizeof(a));
    cin >> n >> m;
    for(int i = 1; i <= n; i++)
    {
        for(int j = 1; j <= m; j++)
        {
            cin >> c;
            if(c == 'S')
            {
                h[1][1] = i;
                h[1][2] = j;
                h[1][3] = 0;
                a[i][j] = false;
            }
            else if(c == 'T')
            {
                fx = i;
                fy = j;
            }
            else if(c == '#') a[i][j] = false;
             
        }
    }
    t = 0, w = 1;
    do
    {
        t++;
        for(int i = 0; i < 4; i++)
        {
            x = h[t][1] + xx[i];
            y = h[t][2] + yy[i];
            if(x > 0 && x <= n && y > 0 && y <= m &&a[x][y])\
            {
                a[x][y] = false;
                w++;
                h[w][1] = x;
                h[w][2] = y;
                h[w][3] = h[t][3] + 1;
                if(x == fx && y == fy)
                {
                    cout << h[w][3] << endl;
                    return 0;
                }
            }
        }
    }while(t < w);
}

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值