图的学习,深度和广度遍历

news2025/1/12 1:55:07

一、什么是图

表示“多对多”的关系
包括:

  • 一组顶点:通常用V(Vertex)表示顶点集合
  • 一组边:通常用E(Edge)表示边的集合
    • 边是顶点对:(v, w)∈E,其中v,w∈V
    • 有向边<v, w>表示从v指向w的边(单行线)
    • 不考虑重边和自回路

在这里插入图片描述
在这里插入图片描述

二、抽象数据类型定义

  • 类型名称:图(Graph)
  • 数据对象集:G(V, E)由一个非空的有限顶点集合v和一个有限边集合E组成。
  • 操作集:对于任意图G ∈ Graph, 以及v ∈ V, e ∈ E
    • Graph Create():建立并返回空图;
    • Graph InsertVertex(Graph G, Vertex v):将v插入G;
    • Graph InsertEdge(Graph G, Edge e):将e插入G;
    • void DFS(Graph G, Vertex v):从顶点v出发深度优先遍历图G;
    • void BFS(Graph G, Vertex v):从顶点v触发宽度优先遍历图G;
    • void ShortestPath(Graph G, Vertex v, int Dist[]):计算图G中顶点v到任一其他顶点的最短距离;
    • void MST(Graph G):计算图G的最小生成树;
  • 数据结构中对于稀疏图的定义为:有很少条边或弧(边的条数|E|远小于|V|²)的图称为稀疏图(sparse graph),反之边的条数|E|接近|V|²,称为稠密图(dense graph)。

如何表示图:
在这里插入图片描述

/* 图的邻接矩阵表示法 */
  
 #define MaxVertexNum 100    /* 最大顶点数设为100 */
 #define INFINITY 65535        /* ∞设为双字节无符号整数的最大值65535*/
 typedef int Vertex;         /* 用顶点下标表示顶点,为整型 */
 typedef int WeightType;        /* 边的权值设为整型 */
 typedef char DataType;        /* 顶点存储的数据类型设为字符型 */
  
 /* 边的定义 */
 typedef struct ENode *PtrToENode;
 struct ENode{
     Vertex V1, V2;      /* 有向边<V1, V2> */
     WeightType Weight;  /* 权重 */
 };
 typedef PtrToENode Edge;
         
 /* 图结点的定义 */
 typedef struct GNode *PtrToGNode;
 struct GNode{
     int Nv;  /* 顶点数 */
     int Ne;  /* 边数   */
     WeightType G[MaxVertexNum][MaxVertexNum]; /* 邻接矩阵 */
     DataType Data[MaxVertexNum];      /* 存顶点的数据 */
     /* 注意:很多情况下,顶点无数据,此时Data[]可以不用出现 */
 };
 typedef PtrToGNode MGraph; /* 以邻接矩阵存储的图类型 */
  
  
  
 MGraph CreateGraph( int VertexNum )
 { /* 初始化一个有VertexNum个顶点但没有边的图 */
     Vertex V, W;
     MGraph Graph;
      
     Graph = (MGraph)malloc(sizeof(struct GNode)); /* 建立图 */
     Graph->Nv = VertexNum;
     Graph->Ne = 0;
     /* 初始化邻接矩阵 */
     /* 注意:这里默认顶点编号从0开始,到(Graph->Nv - 1) */
     for (V=0; V<Graph->Nv; V++)
         for (W=0; W<Graph->Nv; W++)  
             Graph->G[V][W] = INFINITY;
              
     return Graph; 
 }
         
 void InsertEdge( MGraph Graph, Edge E )
 {
      /* 插入边 <V1, V2> */
      Graph->G[E->V1][E->V2] = E->Weight;    
      /* 若是无向图,还要插入边<V2, V1> */
      Graph->G[E->V2][E->V1] = E->Weight;
 }
  
 MGraph BuildGraph()
 {
     MGraph Graph;
     Edge E;
     Vertex V;
     int Nv, i;
      
     scanf("%d", &Nv);   /* 读入顶点个数 */
     Graph = CreateGraph(Nv); /* 初始化有Nv个顶点但没有边的图 */ 
      
     scanf("%d", &(Graph->Ne));   /* 读入边数 */
     if ( Graph->Ne != 0 ) { /* 如果有边 */ 
         E = (Edge)malloc(sizeof(struct ENode)); /* 建立边结点 */ 
         /* 读入边,格式为"起点 终点 权重",插入邻接矩阵 */
         for (i=0; i<Graph->Ne; i++) {
             scanf("%d %d %d", &E->V1, &E->V2, &E->Weight); 
             /* 注意:如果权重不是整型,Weight的读入格式要改 */
             InsertEdge( Graph, E );
         }
     } 
  
     /* 如果顶点有数据的话,读入数据 */
     for (V=0; V<Graph->Nv; V++) 
         scanf(" %c", &(Graph->Data[V]));
  
     return Graph;
 }

领接表:G[N]为指针数组,对应矩阵每行一个链表,只存非0元素。

对于网络,结构中要增加权重的域。

   /* 图的邻接表表示法 */
      
     #define MaxVertexNum 100    /* 最大顶点数设为100 */
     typedef int Vertex;         /* 用顶点下标表示顶点,为整型 */
     typedef int WeightType;        /* 边的权值设为整型 */
     typedef char DataType;        /* 顶点存储的数据类型设为字符型 */
      
     /* 边的定义 */
     typedef struct ENode *PtrToENode;
     struct ENode{
         Vertex V1, V2;      /* 有向边<V1, V2> */
         WeightType Weight;  /* 权重 */
     };
     typedef PtrToENode Edge;
      
     /* 邻接点的定义 */
     typedef struct AdjVNode *PtrToAdjVNode; 
     struct AdjVNode{
         Vertex AdjV;        /* 邻接点下标 */
         WeightType Weight;  /* 边权重 */
         PtrToAdjVNode Next;    /* 指向下一个邻接点的指针 */
     };
      
     /* 顶点表头结点的定义 */
     typedef struct Vnode{
         PtrToAdjVNode FirstEdge;/* 边表头指针 */
         DataType Data;            /* 存顶点的数据 */
         /* 注意:很多情况下,顶点无数据,此时Data可以不用出现 */
     } AdjList[MaxVertexNum];    /* AdjList是邻接表类型 */
      
     /* 图结点的定义 */
     typedef struct GNode *PtrToGNode;
     struct GNode{  
         int Nv;     /* 顶点数 */
         int Ne;     /* 边数   */
         AdjList G;  /* 邻接表 */
     };
     typedef PtrToGNode LGraph; /* 以邻接表方式存储的图类型 */
      
      
      
     LGraph CreateGraph( int VertexNum )
     { /* 初始化一个有VertexNum个顶点但没有边的图 */
         Vertex V;
         LGraph Graph;
          
         Graph = (LGraph)malloc( sizeof(struct GNode) ); /* 建立图 */
         Graph->Nv = VertexNum;
         Graph->Ne = 0;
         /* 初始化邻接表头指针 */
         /* 注意:这里默认顶点编号从0开始,到(Graph->Nv - 1) */
            for (V=0; V<Graph->Nv; V++)
             Graph->G[V].FirstEdge = NULL;
                  
         return Graph; 
     }
             
     void InsertEdge( LGraph Graph, Edge E )
     {
         PtrToAdjVNode NewNode;
          
         /* 插入边 <V1, V2> */
         /* 为V2建立新的邻接点 */
         NewNode = (PtrToAdjVNode)malloc(sizeof(struct AdjVNode));
         NewNode->AdjV = E->V2;
         NewNode->Weight = E->Weight;
         /* 将V2插入V1的表头 */
         NewNode->Next = Graph->G[E->V1].FirstEdge;
         Graph->G[E->V1].FirstEdge = NewNode;
              
         /* 若是无向图,还要插入边 <V2, V1> */
         /* 为V1建立新的邻接点 */
         NewNode = (PtrToAdjVNode)malloc(sizeof(struct AdjVNode));
         NewNode->AdjV = E->V1;
         NewNode->Weight = E->Weight;
         /* 将V1插入V2的表头 */
         NewNode->Next = Graph->G[E->V2].FirstEdge;
         Graph->G[E->V2].FirstEdge = NewNode;
     }
      
     LGraph BuildGraph()
     {
         LGraph Graph;
         Edge E;
         Vertex V;
         int Nv, i;
          
         scanf("%d", &Nv);   /* 读入顶点个数 */
         Graph = CreateGraph(Nv); /* 初始化有Nv个顶点但没有边的图 */ 
          
         scanf("%d", &(Graph->Ne));   /* 读入边数 */
         if ( Graph->Ne != 0 ) { /* 如果有边 */ 
             E = (Edge)malloc( sizeof(struct ENode) ); /* 建立边结点 */ 
             /* 读入边,格式为"起点 终点 权重",插入邻接矩阵 */
             for (i=0; i<Graph->Ne; i++) {
                 scanf("%d %d %d", &E->V1, &E->V2, &E->Weight); 
                 /* 注意:如果权重不是整型,Weight的读入格式要改 */
                 InsertEdge( Graph, E );
             }
         } 
      
         /* 如果顶点有数据的话,读入数据 */
         for (V=0; V<Graph->Nv; V++) 
             scanf(" %c", &(Graph->G[V].Data));
      
         return Graph;
     }

其中

typedef struct Vnode{
    PtrToAdjVNode FirstEdge;/* 边表头指针 */
    DataType Data;            /* 存顶点的数据 */
    /* 注意:很多情况下,顶点无数据,此时Data可以不用出现 */
} AdjList[MaxVertexNum];    /* AdjList是邻接表类型 */

//AdjList是一个Vnode为元素的数组的别名

图的度是和顶点相关联的边的数目

三、图的遍历

3.1 深度优先算法

在这里插入图片描述
邻接表

/* 邻接表存储的图 - DFS*/

void Visit(Vertex V)
{
    printf("Now visit Vertex %d\n", V);
}

/* Visited[]为全局变量,已经初始化false */
void DFS(LGraph Graph, Vertex V, void (*Visit)(Vertex))
{   /* 以V为出发点对邻接表存储的图Graph进行DFS搜索 */
    PtrToAdjVNode W;

    Visit(V);   /* 访问第V个顶点 */
    Visited[V] = true;  /* 标记V已访问 */

    for(W=Graph->G[V].FirstEdge;W;W=W->Next)    /* 对V的每个邻接点W->AdjV */
        if(!Visited[W->AdjV])       /* 若W->AdjV未被访问 */
            DFS(Graph, W->AdjV, Visit);     /* 则递归访问之 */
}

邻接矩阵

void Visit(Vertex V)
{
    printf("Now visit Vertex %d\n", V);
}

void DFS(MGraph Graph, Vertex V, int *Visited)
{
    Vertex W;

    Visit(V);
    Visited[V] = 1;         //已访问

    for(W=0;W<Graph->Nv;W++)
        if(Graph->G[V][W]==1 && Visited[W]==0)
            DFS(Graph, W, Visited);
}

3.2 广度优先算法

邻接矩阵

/* 邻接矩阵存储的图 - BFS */

/* IsEdge(Graph, V, W)检查<V, W>是否图Graph中的一条边,即W是否V的邻接点 */
/* 此函数根据图的不同类型要做不同的实现,关键取决于对不存在的边的表示方法  */
/* 例如对有权图,如果不存在的边被初始化为INFINITY,则函数实现如下:       */
bool IsEdge(MGraph Graph, Vertex V, Vertex W)
{
    return Graph->G[V][W]<INFINITY?true:false;
}

/* Visited[]为全局变量,已经初始化为false */
void BFS(MGraph Graph, Vertex S, void(*Visit)(Vertex))
{   /* 以S为出发点对邻接矩阵存储的图Graph进行BFS搜索 */
    Queue Q;
    Vertex V, W;

    Q = CreateQueue(MaxSize);   /* 创建空队列,MaxSize为外部定义的常数 */
    /* 访问顶点S:此处可根据具体访问需要改写 */
    Visit(S);
    Visited[S] = true;  /* 标记S已访问 */
    AddQ(Q, S);          /* S入对列 */

    while(!IsEmpty(Q)) {
        V = DeleteQ(Q);     /* 弹出V */
        for(W=0;W<Graph->Nv;W++)    /* 对图中的每个顶点W */
            /* 若W是V的邻接点并且未访问过 */
            if(!Visited[W] && IsEdge(Graph, V, W)) {
                /* 访问顶点W */
                Visist(W);
                Visited[W] = true;  /* 标记W已访问 */
                AddQ(Q, W);         /* W入队列 */
            }
    } /* while结束 */
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/981478.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

线性代数的学习和整理20,关于向量/矩阵和正交相关,相似矩阵等

目录 1 什么是正交 1.1 正交相关名词 1.2 正交的定义 1.3 正交向量 1.4 正交基 1.5 正交矩阵的特点 1.6 正交矩阵的用处 1 什么是正交 1.1 正交相关名词 orthogonal set 正交向量组正交变换orthogonal matrix 正交矩阵orthogonal basis 正交基orthogonal decompositio…

【微服务】五. Nacos服务注册

Nacos服务注册 5.1 Nacos服务分级存储模型Nacos服务分级存储模型&#xff1a;服务集群属性&#xff1a;总结&#xff1a; 5.2 根据集群负载均衡总结 5.3 Nacos服务实例的权重设置总结&#xff1a; 5.6 环境隔离namespace总结 5.7 Nacos和Eureka的对比总结 5.1 Nacos服务分级存储…

2023年亲测有效----树莓派启动时自动邮件上报ip

2023年亲测 树莓派启动时自动邮件上报ip 首先开启qq邮箱smtp服务shell文件内容启动自动执行python文件注意事项 首先开启qq邮箱smtp服务 然后点击开启就会有授权码 shell文件内容 在自己的shell里&#xff0c;运行echo $PATH&#xff0c;把内容覆盖下面的path。 功能 作用就…

图解系列 图解Kafka之Producer

开局一张图&#xff0c;其他全靠吹 发送消息流程如下&#xff1a; 1.初始化流程 指定bootstrap.servers&#xff0c;地址的格式为 host:port。它会连接bootstrap.servers参数指定的所有Broker&#xff0c;Producer启动时会发起与这些Broker的连接。因此&#xff0c;如果你为这…

点云从入门到精通技术详解100篇-伪雷达点云预测

前言 近年来,“自动驾驶”已经成为一个耳熟能详的词语,它是一种通过车载计 算实现无人驾驶的智能汽车系统。自动驾驶汽车依靠人工智能、视觉计算、视觉 传感器、控制设备和定位系统协同合作,让系统可以在无人主动操作的情况下, 自动安全地操作机动车辆。其中视觉传感器作…

【继RNN之后的一项技术】Transfomer 学习笔记

谷歌团队在17年的神作&#xff0c;论文17年6月发布 https://arxiv.org/abs/1706.03762 被NIPS2017收录&#xff0c;目前引用量已经逼近3w。 以下内容参考李沐老师的课程《动⼿学深度学习(Pytorch版)》 简介 注意力 自主性&#xff1a;有目的的搜索某样东西&#xff08;键&…

Python小知识 - 如何使用Python进行机器学习

如何使用Python进行机器学习 Python是一种解释型、面向对象、动态数据类型的高级程序设计语言。 机器学习是人工智能的一个分支&#xff0c;是让计算机自动“学习”。学习的过程是从经验E中获得知识K。经验E可以是一个数据集&#xff0c;比如一个图像数据集。知识K可以是计算机…

Python爬虫-爬取文档内容,如何去掉文档中的表格,并保存正文内容

前言 本文是该专栏的第58篇,后面会持续分享python爬虫干货知识,记得关注。 做过爬虫项目的同学,可能或多或少爬取过文档数据,比如说“政务网站,新闻网站,小说网站”等平台的文档数据。爬取文档数据,笔者这里就不过多详述,而本文,笔者将主要介绍在爬取文档数据的过程中…

六安RapidSSL泛域名https能保护几个域名

RapidSSL是Geotrust旗下的子品牌&#xff0c;Geotrust是国际知名的CA认证机构&#xff0c;每年都要通过WebTrust年检&#xff0c;而Geotrust旗下的https证书已经应用于市场多年&#xff0c;Geotrust的子品牌RapidSSL证书经营的DV基础型泛域名https证书也受到市场的欢迎。今天就…

MFC新建内部消息

提示&#xff1a;记录一下MFC新建内部消息的成功过程 文章目录 前言一、pandas是什么&#xff1f;二、使用步骤 1.引入库2.读入数据总结 前言 先说一下基本情况&#xff0c;因为要在mapview上增加一个显示加载时间的功能。然后发现是要等加载完再显示时间&#xff0c;显示在主…

动手学深度学习d2l.Animator无法在PyCharm中显示动态图片的解决方案

from d2l import torch as d2l一、问题描述 运行d2l的训练函数&#xff0c;仅在控制台输出以下内容&#xff0c;无法显示动态图片&#xff08;训练监控&#xff09; <Figure size 350x250 with 1 Axes> <Figure size 350x250 with 1 Axes> <Figure size 350x2…

数据结构与算法之贪心动态规划

一&#xff1a;思考 1.某天早上公司领导找你解决一个问题&#xff0c;明天公司有N个同等级的会议需要使用同一个会议室&#xff0c;现在给你这个N个会议的开始和结束 时间&#xff0c;你怎么样安排才能使会议室最大利用&#xff1f;即安排最多场次的会议&#xff1f;电影的话 那…

Fiddler 系列教程(一)初识Fiddler,我们能用fiddler做什么?

Fiddler是最强大最好用的Web调试工具之一&#xff0c;它能记录所有客户端和服务器的http和https请求&#xff0c;允许你监视&#xff0c;设置断点&#xff0c;甚至修改输入输出数据. 使用Fiddler无论对开发还是测试来说&#xff0c;都有很大的帮助。 阅读目录 Fiddler的基本介…

23个react常见问题

1、setState 是异步还是同步&#xff1f; 合成事件中是异步 钩子函数中的是异步 原生事件中是同步 setTimeout中是同步 相关链接&#xff1a;你真的理解setState吗&#xff1f;&#xff1a; 2、聊聊 react16.4 的生命周期 图片 相关连接&#xff1a;React 生命周期 我对 Reac…

康耐视visionpro破解版满天飞,那么如何查询康耐视Visionpro加密狗支持哪些工具

目录 第一步骤&#xff0c;点击WinR&#xff0c;弹出命令符第二步骤&#xff1a;输入CMD&#xff0c;回车第三步骤&#xff1a;输入cogtool -p&#xff08;cogtool与-p之间有空格&#xff09;&#xff0c;输入完毕后&#xff0c;记得回车&#xff0c;稍等3秒钟不到。 第一步骤&…

[js] 图解 event.pageX event.clientX event.offsetX getBoundingClientRect

event.clientX、event.clientY 鼠标相对于浏览器窗口可视区域的X&#xff0c;Y坐标&#xff08;窗口坐标&#xff09;&#xff0c;可视区域不包括工具栏和滚动条。IE事件和标准事件都定义了这2个属性 event.pageX、event.pageY 类似于event.clientX、event.clientY&#xff0c;…

作为一名游戏开发工作者,分享一些工作经验

作为一名游戏开发工作者&#xff0c;有一些重要的心得和经验可以分享&#xff0c;这些经验可以帮助你更好地在游戏开发领域取得成功&#xff1a; 学习不断进步&#xff1a;游戏开发是一个不断演进的领域&#xff0c;你需要不断学习新的技术和工具。跟踪最新的游戏开发趋势&…

Excel VSTO开发2 -建立Excel VSTO项目

版权声明&#xff1a;本文为博主原创文章&#xff0c;转载请在显著位置标明本文出处以及作者网名&#xff0c;未经作者允许不得用于商业目的。 2 建立Excel VSTO项目 新建项目&#xff0c;选择Excel 2013和2016 VSTO外接程序。输入项目名称&#xff08;本示例的项目名称为&am…

python基于Django健身爱好者交流分享平台vue

而肌友网—健身交流平台能很好地解决这一问题&#xff0c;轻松应对健身交流&#xff0c;既能提高用户对健身交流的评价&#xff0c;又能加快健身交流平台的效率&#xff0c;取代人工管理是必然趋势。 本肌友网—健身交流平台以Django作为框架&#xff0c;B/S模式以及MySql作为后…

Android Studio开发入门教程:如何让开发的app国际化?

配置APP的语言环境&#xff08;文末有Android Studio以及雷电模拟器的压缩包&#xff09; 实验目的&#xff1a; 为了使我们基于android操作系统开发的APP能更好的国际化&#xff0c;面向不同的国家市场&#xff0c;我们需要做出相关的操作&#xff0c;使开发出来的APP会根据用…