服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C/C++ - C语言数据结构与算法之图的遍历(一)

C语言数据结构与算法之图的遍历(一)

2022-07-09 09:29玄澈_ C/C++

这篇文章主要是介绍了利用深度优先算法实现图的遍历,文中利用图文详细的介绍了实现步骤,对我们学习数据结构与算法有一定的帮助,需要的朋友可以参考一下

引入 

在数据结构中常见的有深度优先搜索和广度优先搜索。为什么叫深度和广度呢?其实是针对图的遍历而言的,请看下面这个图:

C语言数据结构与算法之图的遍历(一)

图是由一些小圆点(称为顶点) 和 连接这些点的直线 (称为边)组成的。

例如上图就是由5个顶点(编号为 1,2,3,4,5) 和5条边(1-2,1-3,1-4,2-4)组成。

现在我们从1号顶点开始遍历这个图,遍历就是把图的每一个顶点都访问一次。使用深度优先搜索将会得到如下的结果。

C语言数据结构与算法之图的遍历(一)

图中每个顶点旁边的数表示这个顶点是第几个被访问到的,我们称之为 —— 时间戳 

深度优先搜索

使用深度优先搜索来遍历这个图的过程:

首先从一个未走过的顶点作为起始顶点,比如以1号顶点作为起点。沿1号顶点的边去尝试其他它未走过的顶点,首先发现的是2号顶点还没被走过,于是来到了2号顶点。

再以2号顶点作为出发点继续尝试访问其他未走到过的顶点,这样又来到了4号顶点。

再以4号顶点作为出发点继续尝试访问其他未走过的顶点。但是,此时在4号顶点的周围已经没有其他的顶点了,所以需要返回到2号顶点。返回到2号顶点后,发现沿2号顶点也不能在访问到其他未走到的点了,此时又需要返回到1号顶点。

继续以1号顶点尝试访问其他顶点,我们来到了3号点。以此类推,我们最后来到了5号点。到此,所以的顶点都走过了,遍历结束

深度优先搜索的主要思想是:

首先以一个未被访问的顶点作为起始顶点,沿当前顶点的边走到未被访问过的顶点

当没有未访问过的顶点时,则回到上一个顶点,继续试探访问别的顶点,直到所有的顶点都被访问过。

显然,深度优先搜索是沿着图的某一条分支遍历直至末端,然后回溯,再沿另一条实现相同的遍历,直到所以的顶点都被访问完为止。

代码实现 

C语言数据结构与算法之图的遍历(一)

上面的二维数组中 第i行第j列就是表示顶点i到顶点j是否有边。

1表示有边,x表示没有边,0表示顶点自己到自己。

我们将这种方法称为 ——  图的邻接矩阵储存法。 

细心的朋友可能会发现这张图沿着对角线全部是0,因为上面这张图是 无向图。 

所谓无向图就是指图的边没有方向。例如边 1 - 5 表示 1号顶点可以到 5号顶点,5号顶点也可以到1号顶点。

接下来就是解决怎么用深度优先搜索来实现遍历了:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
void dfs(int cur)               //cur是当前所在的顶点编号
{
    printf("%d", cur);
    sum++;                      //每访问一个点就sum++
    if (sum == n) return;       //所有的顶点都访问过了
    for (i = 1; i <= n; i++) //从1到n的顶点依次尝试,看看有哪些顶点与当前顶点cur有边相连
    {
        //判断当前顶点cur到顶点i是否有边,并判断顶点i是否已被访问过
        {
            if (e[cur][i] == 1 && book[i] == 0)
            {
                book[i] = 1;    //标记顶点i已经访问过
                dfs(i);         //从顶点i出发继续遍历
            }
        }
    }
    return;
}

在上面的代码中 变量 cur 存储的是当前正在遍历的点,二维数组e存储的就是图的边(邻接矩阵),数组book用来标记哪些顶点已经访问过,变量sum用来记录已经访问多少个顶点,变量你存储的是图的顶点总个数。

完整代码 

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
#include <stdio.h>
int book[101], sum, n, e[101][101];
void dfs(int cur)               //cur是当前所在的顶点编号
{
    printf("%d", cur);
    sum++;                      //每访问一个点就sum++
    if (sum == n) return;       //所有的顶点都访问过了
    for (i = 1; i <= n; i++) //从1到n的顶点依次尝试,看看有哪些顶点与当前顶点cur有边相连
    {
        //判断当前顶点cur到顶点i是否有边,并判断顶点i是否已被访问过
        {
            if (e[cur][i] == 1 && book[i] == 0)
            {
                book[i] = 1;    //标记顶点i已经访问过
                dfs(i);         //从顶点i出发继续遍历
            }
        }
    }
    return;
}
 
int main()
{
    int i, j, m, a, b;
    scanf("%d %d", &n, &m);
    //初始化二维矩阵
    for (i = 1; i <= n; i++)
        for (j = 1; j <= n; j++)
            if (i == j) e[i][j] = 0;
            else e[i][j] = 99999999;    //我们假设99999999为x
 
    //读入顶点之间的边
    for (i = 1; i <= n; i++)
    {
        scanf("%d %d", &a, &b);
        e[a][b] = 1;
        e[b][a] = 1;    //因为该图为无向图
    }
 
    //从1号顶点出发
    book[1] = 1;  //标记1号顶点已经访问
    dfs(1);       //从1号顶点开始遍历
 
    return 0;
}

到此这篇关于C语言数据结构与算法之图的遍历(一)的文章就介绍到这了,更多相关C语言数据结构 图的遍历内容请搜索服务器之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持服务器之家!

原文链接:https://blog.csdn.net/forever_bryant/article/details/121723216

延伸 · 阅读

精彩推荐
  • C/C++C++遍历磁盘驱动器的示例代码

    C++遍历磁盘驱动器的示例代码

    这篇文章主要介绍了C++遍历磁盘驱动器的示例代码,帮助大家更好的理解和使用c++,感兴趣的朋友可以了解下...

    凌冷4862021-10-18
  • C/C++C语言实现病例管理系统

    C语言实现病例管理系统

    这篇文章主要为大家详细介绍了C语言实现病例管理系统,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    c7512456304312021-07-20
  • C/C++c++ 防止头文件重复引入的三种方法

    c++ 防止头文件重复引入的三种方法

    这篇文章主要介绍了c++ 防止头文件重复引入的三种方法,帮助大家更好的理解和学习使用c++,感兴趣的朋友可以了解下...

    C语言编程学习基地4912021-10-25
  • C/C++C语言中查询进程信号是否被遮罩或搁置的简单方法

    C语言中查询进程信号是否被遮罩或搁置的简单方法

    这篇文章主要介绍了C语言中查询进程信号是否被遮罩或搁置的简单方法,包括sigprocmask函数和sigpending函数的简介,需要的朋友可以参考下...

    C语言教程网4702021-03-11
  • C/C++一篇文章带你使用C语言编写内核

    一篇文章带你使用C语言编写内核

    内核是操作系统最核心的内容,主要提供硬件抽象层、磁盘及文件系统控制、多任务等功能,由于其涉及非常广泛的计算机知识,很少被人们所熟悉,因而...

    Anita-Sun10332021-12-15
  • C/C++C语言递归操作用法总结

    C语言递归操作用法总结

    这篇文章主要介绍了C语言递归操作用法,结合实例形式总结分析了C语言递归操作的原理、实现技巧与相关应用,需要的朋友可以参考下...

    思齐_4752021-03-25
  • C/C++C语言 全局变量和局部变量详解及实例

    C语言 全局变量和局部变量详解及实例

    这篇文章主要介绍了C语言 全局变量和局部变量详解及实例的相关资料,需要的朋友可以参考下...

    C语言中文网11792021-04-27
  • C/C++C语言实现的统计素数并求和代码分享

    C语言实现的统计素数并求和代码分享

    这篇文章主要介绍了C语言实现的统计素数并求和代码分享,来自PAT平台(浙江大学计算机程序设计能力考试系统)的一个题目,需要的朋友可以参考下...

    C语言教程网9352021-01-30