解题思路:DFS不断向当前结点的下一个结点前进,顺便记下下一个结点的父节点以便回头可以找到所有在环上的结点。用一个数组s[maxn]记录DFS过程中每一个结点的访问状态,对于结点p, 如果:s[p]=-1代表当前结点正在访问中;s[p]=0代表当前未被访问;s[p]=1代表当前结点已经被访问过了。如果在DFS过程中发现当前结点的s[p]=-1代表该结点位于环上,通过父节点回头找就可以找到所有在环上的结点。

注意事项:1.初始结点的父节点初始化为它自己。2.DFS的过程中不要回头找。

参考代码:

#include <iostream>

#include <vector>

#include <algorithm>


using namespace std;


const int maxn = 100000 + 5;

int s[maxn], fa[maxn];

vector<vector<int> > G(maxn);

vector<int> ring;


bool dfs(int p) {

    if (s[p] < 0) {

        int t = p;

        while (fa[t] != p) {

            ring.push_back(t);

            t = fa[t];

        }

        ring.push_back(t);

        return true;

    }

    if (!s[p]) {

        s[p] = -1;

        for (int i = 0; i < G[p].size(); ++i) {

            if (G[p][i] != fa[p]) {

                fa[G[p][i]] = p;

                if (dfs(G[p][i])) {

                    return true;

                }

                fa[G[p][i]] = 0;

            }

        }

        s[p] = 1;

        return false;

    }

    else {

        return false;

    }

}


int main() {

    int N = 0;

    cin >> N;

    int a = 0, b = 0;

    for (int i = 0; i < N; ++i) {

        cin >> a >> b;

        G[a].push_back(b);

        G[b].push_back(a);

    }

    fa[1] = 1;

    dfs(1);

    sort(ring.begin(), ring.end());

    for (int i = 0; i < ring.size(); ++i) {

        cout << ring[i] << ' ';

    }

    return 0;

}




点赞(0)
 

0.0分

0 人评分

C语言网提供由在职研发工程师或ACM蓝桥杯竞赛优秀选手录制的视频教程,并配有习题和答疑,点击了解:

一点编程也不会写的:零基础C语言学练课程

解决困扰你多年的C语言疑难杂症特性的C语言进阶课程

从零到写出一个爬虫的Python编程课程

只会语法写不出代码?手把手带你写100个编程真题的编程百练课程

信息学奥赛或C++选手的 必学C++课程

蓝桥杯ACM、信息学奥赛的必学课程:算法竞赛课入门课程

手把手讲解近五年真题的蓝桥杯辅导课程

评论列表 共有 0 条评论

暂无评论