c语言数据结构-图的遍历

news/2024/7/15 17:32:18 标签: 数据结构, c语言, 图论, 深度优先, 广度优先

 (创作不易,感谢有你,你的支持,就是我前行的最大动力,如果看完对你有帮助,请留下您的足迹) 

目录

定义:

两种遍历方法: 

深度优先搜索(DFS):

广度优先搜索(BFS):

定义:

从已给的连通图中某一顶点出发,沿着一些边访遍图中所有的顶点,且使每个顶点仅被访
问一次,就叫做图的遍历 

图遍历的实质是找每个顶点的邻接点的过程

                                          

特点:图中可能存在回路 ,且图的任一顶点都可能与其它顶点相通,在访问完某个顶点之后可能会沿着某些边 又回到了曾经访问过的顶点 

两种遍历方法: 

深度优先搜索(DFS):

仿树的前序遍历过程 (http://t.csdn.cn/zVDgK)

 

如图所示:

1.访问图中某一起始顶点V1,由V1出发,访问它的任一邻接顶点V2

2.再从V2出发,访问与V2邻接但还未被访问过的顶点V3

3.然后再从V3出发,进行类似的访问......

4.如此进行下去,直至到达所有的邻接顶点都被访问过的顶点V5为止

5,接着退回一步,退到前一次刚访问过的顶点,看是否还有其它没有被访问的邻接顶点
如果没有就再退回一步进行搜索,之后再从此顶点出发,进行与前述类似的访问,
如果有则访问此顶点
6.重复上述过程,直到连通图中所有顶点都被访问过为止

/** 保存每个顶点的访问状态[0-未访问;1-已访问 */
int visited[100];

/** 图的类型枚举 */
typedef enum
{
    DG,     //有向图   0
    UDG,    //无向图   1
    DN,     //有向网   2
    UDN     //无向网   3
}GraphKind;

/** 图的邻接矩阵存储表示 */
typedef struct
{
    char* verTexs[100];                     //顶点数组
    int arcs[100][100];                     //邻接矩阵(权数组)
    int verTexCount;                        //图的顶点数
    int arcCount;                           //图的边/弧数
    GraphKind kind;                         //图的类型
}MatrixGraph;

/** 返回某个顶点在顶点集合中的下标(从0开始),不存在返回-1 */
int LocateVex(MatrixGraph* G, char* vex) {
    int index = 0;
    while (index < G->verTexCount) {
        if (strcmp(G->verTexs[index], vex) == 0) {
            break;
        }
        index++;
    }
    return index == G->verTexCount ? -1 : index;
}

/** 深度优先搜索的核心算法,index为深度搜索的某个顶点下标 */
void DFS_AMG(MatrixGraph G, int index) {
    printf(" -> %s", G.verTexs[index]);     //访问当前顶点
    visited[index] = 1;               //更改当前顶点的访问状态
    for (int i = FirstAdjVex_AMG(G, G.verTexs[index]); i;
        i = SecondAdjVex_AMG(G, G.verTexs[index], G.verTexs[i])) {
        if (!visited[i]) {
            DFS_AMG(G, i);  //如果没有访问过,就继续递归调用访问
        }
    }
}
/** 返回顶点vex所在行中的第一个邻接点下标 */
int FirstAdjVex_AMG(MatrixGraph G, char * vex) {
    int i = LocateVex(&G, vex);         //找到顶点vex在顶点数组中的下标
    if (i == -1) return 0;
    int defaultWeight;                  //默认权重
    defaultWeight = G.kind <= 1 ? 0 : INT_MAX;          //图/网
    //搜索图的邻接矩阵中与顶点vex的第一个邻接点下标
    for (int j = i + 1; j < G.verTexCount; j++) {
        if (G.arcs[i][j] != defaultWeight) {
            return j;
        }
    }
    return 0;
}

/** 返回与顶点vex1邻接的另一个邻接点,没有就返回0 */
int SecondAdjVex_AMG(MatrixGraph G, char * vex1, char * vex2) {
    int index1 = LocateVex(&G, vex1);
    int index2 = LocateVex(&G, vex2);
    if (index1 == -1 || index2 == -1) return 0;
    int defaultWeight;
    defaultWeight = G.kind <= 1 ? 0 : INT_MAX;
    for (int i = index2 + 1; i < G.verTexCount; i++) {
        if (G.arcs[index1][i] != defaultWeight) {
            return i;
        }
    }
    return 0;
}

/** 邻接矩阵的深度优先遍历 */
void DFSTraverse_AMG(MatrixGraph G) {
    //初始化状态数组
    for (int i = 0; i < G.verTexCount; i++) {
        visited[i] = 0;     //初始状态设置为未访问
    }
    //DFS遍历
    for (int i = 0; i < G.verTexCount; i++) {
        if (!visited[i]) {//如果某个顶点未访问
            //调用遍历函数
            DFS_AMG(G, i);
        }
    }
}

广度优先搜索(BFS):

广度优先搜索是一种 分层 的搜索过程,每向前走一步可能访问一批顶点,不像深度优先搜索那样有回退的情况。因此,广度优先搜索 不是一个递归的过程

        

 如图所示:

1.访问起始点V1

2.依次访问V1的邻接点

3.依次访问这些顶点中未被访问过的邻接点

4.直到所有顶点都被访问过为止

#include"队列.cpp"
/** 保存每个顶点的访问状态[0-未访问;1-已访问 */
int visited[100];

/** 图的类型枚举 */
typedef enum
{
    DG,     //有向图   0
    UDG,    //无向图   1
    DN,     //有向网   2
    UDN     //无向网   3
}GraphKind;

/** 图的邻接矩阵存储表示 */
typedef struct
{
    char* verTexs[100];                     //顶点数组
    int arcs[100][100];                     //邻接矩阵(权数组)
    int verTexCount;                        //图的顶点数
    int arcCount;                           //图的边/弧数
    GraphKind kind;                         //图的类型
}MatrixGraph;

/** 返回某个顶点在顶点集合中的下标(从0开始),不存在返回-1 */
int LocateVex(MatrixGraph* G, char* vex) {
    int index = 0;
    while (index < G->verTexCount) {
        if (strcmp(G->verTexs[index], vex) == 0) {
            break;
        }
        index++;
    }
    return index == G->verTexCount ? -1 : index;
}

/** 返回顶点vex所在行中的第一个邻接点下标 */
int FirstAdjVex_AMG(MatrixGraph G, char* vex) {
    int i = LocateVex(&G, vex);         //找到顶点vex在顶点数组中的下标
    if (i == -1) return 0;
    int defaultWeight;                  //默认权重
    defaultWeight = G.kind <= 1 ? 0 : INT_MAX;          //图/网
    //搜索图的邻接矩阵中与顶点vex的第一个邻接点下标
    for (int j = i + 1; j < G.verTexCount; j++) {
        if (G.arcs[i][j] != defaultWeight) {
            return j;
        }
    }
    return 0;
}

/** 返回与顶点vex1邻接的另一个邻接点,没有就返回0 */
int SecondAdjVex_AMG(MatrixGraph G, char* vex1, char* vex2) {
    int index1 = LocateVex(&G, vex1);
    int index2 = LocateVex(&G, vex2);
    if (index1 == -1 || index2 == -1) return 0;
    int defaultWeight;
    defaultWeight = G.kind <= 1 ? 0 : INT_MAX;
    for (int i = index2 + 1; i < G.verTexCount; i++) {
        if (G.arcs[index1][i] != defaultWeight) {
            return i;
        }
    }
    return 0;
}

/** 广度优先搜索的核心算法 - index为广度优先搜索的某个顶点下标 */
void BFS_AMG(MatrixGraph G, int index) {
    printf(" -> %s", G.verTexs[index]);
    visited[index] = 1;               //设置顶点状态为已访问

    SeqQueue queue;
    initseqqueue(&queue);
    //将当前顶点入队
    OfferSeqQueue(&queue, G.verTexs[index]);
    while (queue.front != queue.rear) {
        //取出队头元素,遍历队头顶点的所有邻接点
        char* vex;                     //取出的队头顶点
        PollSeqQueue(&queue, &vex);
        for (int i = FirstAdjVex_AMG(G, vex); i; i = SecondAdjVex_AMG(G, vex, G.verTexs[i])) {
            if (!visited[i]) {
                printf(" -> %s", G.verTexs[i]);
                visited[i] = 1;        //设置顶点状态为已访问
                OfferSeqQueue(&queue, G.verTexs[i]);
            }
        }
    }
}

/** 邻接矩阵的广度优先遍历 */
void BFSTraverse_AMG(MatrixGraph G) {
    //初始化状态数组
    for (int i = 0; i < G.verTexCount; i++) {
        visited[i] = 0;
    }
    //循环遍历每个顶点
    for (int i = 0; i < G.verTexCount; i++) {
        if (!visited[i]) {
            BFS_AMG(G, i);
        }
    }
}

 


http://www.niftyadmin.cn/n/56534.html

相关文章

七、插件机制

Interceptor MyBatis 插件模块中最核心的接口就是 Interceptor 接口&#xff0c;它是所有 MyBatis 插件必须要实现的接口&#xff0c;其核心定义如下&#xff1a; public interface Interceptor {// 插件实现类中需要实现的拦截逻辑Object intercept(Invocation invocation) …

ffmpeg转码转封装小工具开发

如下图所示&#xff0c;是本人开发的一个转码转封装小工具 其中目标文件视频编码格式支持&#xff1a;H264&#xff0c;H265&#xff0c;VP8&#xff0c;VP9。 目标文件封装格式支持&#xff1a;mp4,mkv,avi,mov,flv。 目标文件音频编码格式支持两个&#xff0c;COPY和AAC&am…

力扣sql简单篇练习(十五)

力扣sql简单篇练习(十五) 1 直线上的最近距离 1.1 题目内容 1.1.1 基本题目信息 1.1.2 示例输入输出 1.2 示例sql语句 SELECT min(abs(p1.x-p2.x)) shortest FROM point p1 INNER JOIN point p2 ON p1.x <>p2.x1.3 运行截图 2 只出现一次的最大数字 2.1 题目内容 2…

NFC 项目前期准备工作

同学,别退出呀,我可是全网最牛逼的 WIFI/BT/GPS/NFC分析博主,我写了上百篇文章,请点击下面了解本专栏,进入本博主主页看看再走呗,一定不会让你后悔的,记得一定要去看主页置顶文章哦。 了解项目信息,FAE联系方式,驱动源码等驱动合入内核配置DTS驱动设备节点验证Push nf…

Python 之 NumPy 简介和创建数组

文章目录一、NumPy 简介1. 为什么要使用 NumPy2. NumPy 数据类型3. NumPy 数组属性4. NumPy 的 ndarray 对象二、numpy.array() 创建数组1. 基础理论2. 基础操作演示3. numpy.array() 参数详解三、numpy.arange() 生成区间数组四、numpy.linspace() 创建等差数列五、numpy.logs…

初学者必读:讲解 VC 下如何正确的创建、管理及发布项目

Visual C 的项目文件组成&#xff0c;以及如何正确的创建及管理项目。 本内容是初学者必须要掌握的。不能正确的管理项目&#xff0c;就不能进一步写有规模的程序。 一、项目下各种常见文件类型的功能 1. 代码文件 扩展名为 .cpp、.c、.h 等。 通常情况下&#xff0c;项目…

php的declare命令如何使用?

php中的declare结构用来设定一段代码的执行指令declare用于执行3个指令&#xff1a;ticks,encoding,strict_typesdeclare结构用于全局范围&#xff0c;影响到其后的所有代码&#xff08;但如果有declare结构的文件被其他文件包含&#xff0c;则对包含他的父文件不起作用&#x…

BiseNet v1论文及其代码详解

来源&#xff1a;投稿 作者&#xff1a;蓬蓬奇 编辑&#xff1a;学姐 BiSeNet v1说明&#xff1a; 文章链接&#xff1a;https://arxiv.org/abs/1808.00897 官方开源代码&#xff1a;https://github.com/CoinCheung/BiSeNet &#xff08;本文未使用&#xff09; 文章标题&am…