当前位置:   article > 正文

图论之邻接矩阵

邻接矩阵

路径规划系列文章目录

  1. 路径规划算法综述
  2. 图论基础介绍

目录

路径规划系列文章目录

一、图的存储方式介绍

二、邻接矩阵介绍

三、邻接矩阵实现

四、总结


一、图的存储方式介绍

         图的结构比较复杂,是非线性结构,任意两点都可能存在联系,相对来说存储方法较多。目前主要有:

  1. 邻接矩阵表示法
  2. 邻接表表示法
  3. 邻接多重表表示法
  4. 十字链表表示法

        无论上述哪种存储方式,我们都要存顶点的信息,在本系列文章中,我们介绍1,2两种表示法。

二、邻接矩阵介绍

        邻接矩阵就是利用二维矩阵表示图中各顶点之间的关系,对于有n个顶点的图来说,用n阶方阵来表示该图,其中矩阵元素A_{ij}表示从顶点v_{i}v_{j}之间的边,A_{ij}的大小表示边的权值。如果顶点v_{i}v_{j}没有边,则可以将A_{ij}设置为0或者\infty

        如下图所示,左边是一个无向图,右边是其对应的邻接矩阵,该图是无权图,因此有边的值都设置为1。

                           

        下面是有向图及其邻接矩阵

                                     

         从上面可见,无向图的邻接矩阵是关于主轴对称的,第i行或第j列就是顶点v_{i}的度(边数)。图中的边数为"1的个数"/2。对于有向图,由于其具有方向性,因此邻接矩阵一般是不对称的第i行1的个数是顶点v_{i}的出度第i列1的个数是其入度。图的边数等于矩阵中1的个数

对于带权图来说,只需要将1替换为边的权值即可,下面是带权图及其邻接矩阵。

                          

        其中,\infty表示没有边,可以是一个计算机能够接受的较大的值即可。

三、邻接矩阵实现

  1. #include<iostream>
  2. using namespace std;
  3. #define INF 65535 //表示无穷大,其他合理的值也可
  4. #define MaxVerNum 1000 //定义顶点最大数量
  5. typedef int cellType; //定义邻接矩阵元素数据类型,即权值的数据类型
  6. //定义图的类型分别为无向图,无向带权图,有向图,有向带权图
  7. typedef enum{
  8. UDG,UDN,DG,DN
  9. }GraphKind;
  10. class GraphAdjMatrix
  11. {
  12. private:
  13. int VerNum;//顶点数量
  14. int ArcNum;//边数量
  15. GraphKind gKind; //图类型
  16. cellType** AdjMatrix;//邻接矩阵
  17. public:
  18. GraphAdjMatrix();
  19. void createGraph();//构建图
  20. void GraphSet(int VerNum,int ArcNum,int kind);//图属性设置
  21. int getVerNum() {return VerNum;}
  22. int getArcNum() {return ArcNum;}
  23. GraphKind geyGraphKind() {return gKind;};
  24. void setMatrix(int i,int j,int w) ; //邻接矩阵设置
  25. void printMatrix();//打印邻接矩阵
  26. };
  27. GraphAdjMatrix::GraphAdjMatrix()//构造函数
  28. {
  29. AdjMatrix = new cellType*[MaxVerNum];
  30. for(int i=0;i<MaxVerNum;i++)// 为邻接矩阵分配内存
  31. AdjMatrix[i] = new cellType[MaxVerNum];
  32. }
  33. void GraphAdjMatrix::setMatrix(int i,int j,int w)
  34. {
  35. AdjMatrix[i][j]=w;
  36. if(gKind==UDG||gKind==UDN) //如果是无向图,则设置对称位置权重
  37. AdjMatrix[j][i]=w;
  38. };
  39. void GraphAdjMatrix::createGraph()
  40. {
  41. int vn,an,k;//分别代表顶点数量,边数量,以及图类型
  42. cout<<"输入顶点数量,边数量,图类型用空格隔开"<<endl;
  43. cout<<"0-无向无权图 1-无向带权图 2-有向无权图 3-有向带权图"<<endl;
  44. cin>>vn>>an>>k;
  45. VerNum = vn;
  46. ArcNum = an;
  47. gKind = (GraphKind)k;
  48. int i,j,w;
  49. //初始化邻接矩阵
  50. for(int i=1;i<=vn;i++)
  51. {
  52. for(int j=1;j<=vn;j++)
  53. {
  54. AdjMatrix[i][j]=INF;
  55. }
  56. }
  57. /*无向图,无向带权图,有向图,有向带权图 */
  58. GraphKind gk;
  59. while(an--)
  60. {
  61. if (k == UDG || k == DG)//如果是无权图,则将边权重设为1
  62. {
  63. cin>>i>>j;
  64. AdjMatrix[i][j]=1;
  65. if (k==UDG)//如果是无向图,对称位置设置权重
  66. AdjMatrix[j][i]=1;
  67. }
  68. else
  69. {
  70. cin>>i>>j>>w;
  71. AdjMatrix[i][j]=w;
  72. if (k == UDN)//如果是无向图,对称位置设置权重
  73. AdjMatrix[j][i]=w;
  74. }
  75. }
  76. }
  77. void GraphAdjMatrix::printMatrix()
  78. {
  79. for(int i=1;i<=VerNum;i++)
  80. {
  81. for(int j=1;j<=VerNum;j++)
  82. {
  83. if (AdjMatrix[i][j]==INF)
  84. cout<<"*"<<"\t";
  85. else
  86. cout<<AdjMatrix[i][j]<<"\t";
  87. }
  88. cout<<endl;
  89. }
  90. }
  91. int main()
  92. {
  93. GraphAdjMatrix cg;
  94. cg.createGraph();
  95. cg.printMatrix();
  96. return 0;
  97. }

四、总结

        图的邻接矩阵表示的优点: 非常直观,并且容易实现,编写算法也较简便,因而应用较广; 根据矩阵元素Aij=1或0,便于判定两个顶点之间是否有边(弧)相连; 计算顶点的度数,或有向图的入度、出度方便; 计算图的边数算法简单等。

        图的邻接矩阵表示的缺点: 邻接矩阵事实上是一种顺序存储结构,具有顺序结构共有的缺点,比如:只能按最大空间需求申请内存空间、插入和删除顶点复杂等; 空间复杂度高,n个顶点的图,存储邻接矩阵需要n2个单元,如果一个图的顶点数较多,但边(弧)数较少的话--稀疏图,邻接矩阵一样需要n2个存储单元,就太浪费存储空间; 统计图的边数算法虽然简单,用双重循环统计“1”的个数即可,但其时间复杂度为O(n2)。

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/煮酒与君饮/article/detail/874019
推荐阅读
相关标签
  

闽ICP备14008679号