博主头像
mxd's Blog

"The quieter you become,the more you are able to hear."

C++ 深度优先搜索 课堂笔记 + 例题题解

深度优先搜索

DFS是可用于遍历树或者图的搜索算法,DFS与回溯法类似,一条路径走到底后需要返回上一步,搜索第二条路径。在树的遍历中,首先一直访问到最深的节点,然后回溯到它的父节点,遍历另一条路径,直到遍历完所有节点。图也类似,如果某个节点的邻居节点都已遍历,回溯到上一个节点。

深度优先搜索是图论中的经典算法,利用深度优先搜索算法可以产生目标图的相应拓扑排序表,利用拓扑排序表可以方便的解决很多相关的图论问题,如最大路径问题等等。一般用栈数据结构来辅助实现DFS算法。根据深度深度优先搜索的特点,采用递归函数实现比较简单。但也可以不采用递归。

例题 【题解】

本文代码严格按照Linux C++标准编写,空格、缩进符合标准。

1.[入门] 扫地机器人

// [入门] 扫地机器人
// Author:mxdyeah Date:20240529
#include <bits/stdc++.h>
using namespace std;
const int maxn = 101;
int n, m;
int a[maxn][maxn];
int fx[5] = {0, 0, 1, 0, -1};
int fy[5] = {0, 1, 0, -1, 0};
void p1554(int x, int y, int k)
{
    a[x][y] = k;
    int tx, ty;
    for (int i = 1; i <= 4; i++) {
        tx = x + fx[i];
        ty = y + fy[i];
        if (tx >= 1 and tx <= n and ty >= 1 and ty <= m and a[tx][ty] == 0) {
            p1554(tx, ty, k + 1);
        }
    }
}
int main()
{
    cin >> n >> m;
    p1554(1, 1, 1);
    int i, j;
    for (i = 1; i <= n; i++) {
        for (j = 1; j <= m; j++) {
            if (a[i][j] < 10) {
                cout << " ";
            }
            cout << " ";
            cout << a[i][j];
        }
        cout << endl;
    }
    return 0;
}

2.【基础】迷宫出口

// 【基础】迷宫出口
// Author:mxdyeah Date:20240529
#include<bits/stdc++.h>
using namespace std;
int a[110][110];
int n, s1, s2, e1, e2;
bool f = false;
void mxdyeah(int x, int y)
{
    int fx[5] = {0, 0, 1, 0, -1};
    int fy[5] = {0, 1, 0, -1, 0};
    a[x][y] = 1;
    int tx, ty;
    for (int i = 1; i <= 4; i++) {
        tx = x + fx[i];
        ty = y + fy[i];
        if (tx >= 1 and tx <= n and ty >= 1 and ty <= n and a[tx][ty] == 0) {
            if (tx == e1 && ty == e2) {
                f = true;
            }
            else {
                mxdyeah(tx, ty);
            }
        }
    }
}
int main()
{
    cin >> n;
    for (int i = 1; i <= n; i++) {
        for (int j = 1; j <= n; j++) {
            cin >> a[i][j];
        }
    }
    cin >> s1 >> s2 >> e1 >> e2;
    if (a[s1][s2] == 1 or a[e1][e2] == 1) {
        cout << "NO";
    }
    else {
        mxdyeah(s1, s2);
        if (f == true) {
            cout << "YES";
        }
        else {
            cout << "NO";
        }
    }
    return 0;
}

3.数池塘(四方向) USACO

// <pre class="prism-highlight" prism-language-cpp="" blog="mxdblog_zblog">
// Author:mxdyeah Date:20240529
#include<bits/stdc++.h>
using namespace std;
int n, m, ans;
int dx[4] = {0, 0, 1, -1}, dy[4] = {1, -1, 0, 0};
char a[105][105];
int vis[105][105];
void dfs(int xx, int yy)
{
    vis[xx][yy] = 1;
    for (int i = 0; i < 4; i++) {
        int x = xx + dx[i], y = yy + dy[i];
        if (x >= 0 && y >= 0 && x < n && y < m && !vis[x][y] && a[x][y] == 'W') {
            dfs(x, y);
        }
    }
}
int main()
{
    scanf("%d%d", &n, &m);
    for (int i = 0; i < n; i++) {
        cin >> a[i];
    }
    for (int i = 0; i < n; i++) {
        for (int j = 0; j < m; j++) {
            if (!vis[i][j] && a[i][j] == 'W') {
                ans++;
                dfs(i, j);
            }
        }
    }
    cout << ans << endl;
    return 0;
}

4.数池塘(八方向)

//Author:mxdyeah
#include <bits/stdc++.h>
using namespace std;
int fx[9] = {0, 0, 1, 1, 1, 0, -1, -1, -1};
int fy[9] = {0, 1, 1, 0, -1, -1, -1, 0, 1};
int n, m, cnt = 0;
char a[110][110];
void mxd(int x, int y)
{
    a[x][y] = '.';
    int tx, ty;
    for (int i = 1 ; i <= 8 ; i++) {
        tx = x + fx[i];
        ty = y + fy[i];
        if (a[tx][ty] == 'W')
            mxd(tx, ty);
    }
}
int main()
{
    scanf("%d%d", &n, &m);
    for (int i = 1 ; i <= n ; i++)
        for (int j = 1 ; j <= m ; j++)
            cin >> a[i][j];
    for (int i = 1 ; i <= n ; i++)
        for (int j = 1 ; j <= m ; j++)
            if (a[i][j] == 'W') {
                cnt++;
                mxd(i, j);
            }
    printf("%d", cnt);
    return 0;
}

5.奶牛和草丛 USACO

//Author:mxdyeah
# include <bits/stdc++.h>
using namespace std;
int n, m;
char a[104][104];
long long num;
void dg(int x, int y)
{
    if (x < 1 || x > n || y < 1 || y > m)
        return ;
    if (a[x][y] == '#') {
        a[x][y] = '.';
        dg(x + 1, y);
        dg(x - 1, y);
        dg(x, y + 1);
        dg(x, y - 1);
    }
    return ;
}
int main()
{
    cin >> n >> m;
    for (int u = 1; u <= n; u++) {
        for (int v = 1; v <= m; v++) {
            cin >> a[u][v];
        }
    }
    for (int u = 1; u <= n; u++) {
        for (int v = 1; v <= m; v++) {
            if (a[u][v] == '#') {
                dg(u, v);
                num += 1;
            }
        }
    }
    cout << num;
    return 0;
}
C++ 深度优先搜索 课堂笔记 + 例题题解
https://blog.mxdyeah.top/mxdyeah_blog_post/53.html
本文作者 mxdyeah
发布时间 2024-05-29
许可协议 CC BY-NC-SA 4.0
发表新评论