玩转图论算法C#版笔记02:图的深度优先遍历

本文介绍了图的深度优先遍历算法,通过对比二叉树的前序遍历伪代码进行讲解,并展示了递归过程。接着,详细探讨了深度优先遍历在求无向图连通分量、路径问题(包括单源路径)、无向图环检测和二分图检测中的应用。提供了相应的代码实现,帮助理解算法的时间复杂度和实际应用。
摘要由CSDN通过智能技术生成

目录

1 图的深入优先遍历伪代码及递归演示

1.1 二叉树前序遍历与图的前序遍历伪代码对比

1.2 图的深度优先遍历递归过程推演

1.3 前序/后序遍历代码实现

2 图的深度优先遍历应用及代码实现

2.1 求无向图中的连通分量

2.2 路径问题

2.2.1 单源路径问题

2.2.2 路径问题

2.3 无向图环检测

2.4 二分图检测


1 图的深入优先遍历伪代码及递归演示

前置准备:

回顾LeetCode 144/94/145/102号题目,二叉树的前序遍历、中序遍历、后序遍历和层序遍历

回顾589/590/429号题目,N叉树的前序遍历、后序遍历和层序遍历

1.1 二叉树前序遍历与图的前序遍历伪代码对比

先来一段算法的伪代码:

二叉树的深度优先遍历 图的深度优先遍历
preorderTraversal(root) dfs(v)
            proorder(TreeNode node)
            {
                if(node != null)
                {
                    List.add(node.val);
                    preorder(node.left);
                    preorder(node.right);
                }
            }
            bool[] visited = new bool[V];
            preorder(int v)
            {
                visited[v] = 1;
                List.add[v];
                foreach(int w in adjH[v])
                {
                    if(!visited[w])
                    {
                        dfs[w];
                    }
                }
            }

1.2 图的深度优先遍历递归过程推演

 

深度优先遍历执行过程为:

根据上述顺序,确定的遍历结果为:0-1-3-2-6-5-4

1.3 前序/后序遍历代码实现

后续优先遍历在有向图中有应用。

代码实现:

其中Graph类为上一篇博客文章中基于SortedSet实现的邻接表类,更换类名得到的。

    class GraphDFS
    {
        Graph G;
        private bool[] visit;//记录节点是否访问的数组
        private List<int> pre; //前序遍历结果存储
        private List<int> post; //后序遍历结果存储

        public GraphDFS(Graph graph)
        {
            this.G = graph;
            visit = new bool[G.VertexCount];
            pre = new List<int>();
            post = new List<int>();

            //循环是为了处理存在多个联通分量的情形
            for(int v = 0; v < G.EdgeCount; v++)
            {
                if(!visit[v])
                {
                    dfs(v);
                }
            }
        }

        /// <summary>
        /// 图的深度优先遍历
        /// </summary>
        /// <param name="v">要遍历的节点</param>
        private void dfs(int v)
        {
            visit[v] = true;
            pre.Add(v);
            foreach (int w in G.getAdjEdge(v))
            {
                if (!visit[w])
                {
                    dfs(w);
                }
            }
            post.Add(v);
        }

        /// <summary>
        /// 获取前序遍历列表
        /// </summary>
        /// <returns></returns>
        public List<int> getPreResault()
        {
            return pre;
        }
        /// <summary>
        /// 获取后序遍历列表
        /// </summary>
        /// <returns></returns>
        public List<int> getPostResault()
        {
            return post;
        }

        /// <summary>
        /// 打印遍历结果
        /// </summary>
        /// <param name="isPre">是否为前序遍历</param>
        public void printResault(bool isPre)
        {
            if(isPre)
            {
                Console.Write("前序遍历结果:");
                foreach(int v in pre)
                {
                    Console.Write(s
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小薛引路

喜欢的读者,可以打赏鼓励一下

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值