文章目录
- 一、定义
- 二、结构
- 三、常用操作
- 结语
- 附录
一、定义
图的邻接矩阵是一种采用邻接矩阵数组表示顶点之间相邻关系的存储结构。设图G有n个顶点,则邻接矩阵是一个n*n的方阵,定义为:
下面给出一个例子:
一个有向图G1和一个无向图G2
将上诉的有向图和无向图用邻接矩阵表示
二、结构
图示
代码描述
//设置默认的顶点个数
#define Default_Vertex_Size 10
//数据类型
#define T char
//邻接矩阵图结构
typedef struct GraphMtx
{
int MaxVertices; //最大顶点数
int NumVertices; //真实的顶点数
int NumEdges; //边数
T *VerticesList; //顶点列表
int **Edge; //边信息矩阵
}GraphMtx;
三、常用操作
下面给出无向图的操作实现(有向图方法类似)。
初始化
//图的初始化
void InitGraph(GraphMtx *g)
{
g->MaxVertices = Default_Vertex_Size;//最大顶点数初始化
g->NumVertices = g->NumEdges = 0; //实际顶点数初始化
//分配顶点存储列表的空间
g->VerticesList = (T*)malloc(sizeof(T)*(g->MaxVertices));
assert(g->VerticesList != NULL);
//开辟边信息存储矩阵的空间(二维数组的动态开辟)
g->Edge = (int**)malloc(sizeof(int*) * g->MaxVertices);//总行数的开辟
assert(g->Edge != NULL);
for(int i=0; i<g->MaxVertices; ++i)//每一行内具体的空间开辟
{
g->Edge[i] = (int*)malloc(sizeof(int) * g->MaxVertices);
}
for(i=0; i<g->MaxVertices; ++i) //初始化
{
for(int j=0; j<g->MaxVertices; ++j)
{
g->Edge[i][j] = 0;
}
}
}
获取顶点位置
//获取顶点的位置
int GetVertexPos(GraphMtx *g, T v)
{
for(int i=0; i<g->NumVertices; ++i) //对所有顶点进行遍历
{
//判断是否找到顶点v所在位置
if(g->VerticesList[i] == v)
return i;
}
return -1;
}
打印图信息
//打印图信息
void ShowGraph(GraphMtx *g)
{
printf(" ");
for(int i=0; i<g->NumVertices; ++i) //获取顶点,并打印
{
printf("%c ",g->VerticesList[i]);
}
printf("\n");
for(i=0; i<g->NumVertices; ++i) //打印顶点间边的信息
{
printf("%c ",g->VerticesList[i]);
for(int j=0; j<g->NumVertices; ++j)
{
printf("%d ",g->Edge[i][j]);
}
printf("\n");
}
printf("\n");
}
插入顶点
//插入顶点
void InsertVertex(GraphMtx *g, T v)
{
if(g->NumVertices >= g->MaxVertices) //判断顶点空间是否已满
return;
g->VerticesList[g->NumVertices++] = v; //还有空间,放入顶点
}
插入边
//插入边:在v1和v2顶点间插入边
void InsertEdge(GraphMtx *g, T v1, T v2)
{
int p1 = GetVertexPos(g,v1); //获取v1顶点位置
int p2 = GetVertexPos(g,v2); //获取v2顶点位置
if(p1==-1 || p2==-1)
return;
if(g->Edge[p1][p2] != 0)
return;
//无向图存储 需要双向的
g->Edge[p1][p2] = g->Edge[p2][p1] = 1;
g->NumEdges++; //记录实际边数
}
删除边
//删除边:删除v1和v2顶点间的边
void RemoveEdge(GraphMtx *g, T v1, T v2)
{
//求出两个顶点的下标位置
int p1 = GetVertexPos(g,v1);
int p2 = GetVertexPos(g,v2);
if(p1==-1 || p2==-1)
return;
if(g->Edge[p1][p2] == 0)
return;
//将边清空
g->Edge[p1][p2] = g->Edge[p2][1] = 0;
g->NumEdges--; //更新边数
}
删除顶点
//删除顶点
void RemoveVertex(GraphMtx *g, T v)
{
//获取顶点的位置
int p = GetVertexPos(g,v);
if(p == -1)
return;
//释放顶点
int numedges = 0;
for(int i=p; i<g->NumVertices-1; ++i)
{//将要释放顶点之后的顶点逐一前移
g->VerticesList[i] = g->VerticesList[i+1];
}
//统计与要删除顶点相连的边条数
for(i=0; i<g->NumVertices; ++i)
{
if(g->Edge[p][i] != 0)
{
numedges++;
}
}
//删除与释放顶点相连的边(更改存放边信息的矩阵)
for(i=p; i<g->NumVertices-1; ++i)//删除行
{//将要删除行之后的行逐一向前移动一行
for(int j=0; j<g->NumVertices; ++j)
{
g->Edge[i][j] = g->Edge[i+1][j];
}
}
for(i=p; i<g->NumVertices; ++i)//删除列
{//将要删除列之后的列逐一向前移动一列
for(int j=0; j<g->NumVertices; ++j)
{
g->Edge[j][i] = g->Edge[j][i+1];
}
}
g->NumVertices--;
g->NumEdges -= numedges;
}
获取v第一个邻接顶点
//获取v第一个邻接顶点
int GetFirstNeighbor(GraphMtx *g, T v)
{
//获取顶点v所在位置
int p = GetVertexPos(g,v);
if(p == -1)
return -1;
//对顶点进行搜索,看那个顶点与v相连
for(int i=0; i<g->NumVertices; ++i)
{
if(g->Edge[p][i] == 1)//判断是否,找到
return i;//找到即返回
}
return -1;
}
获取下一个邻接顶点
//获取下一个邻接顶点:获取顶点v下一个邻接顶点w的邻接顶点
int GetNextNeighbor(GraphMtx *g, T v, T w)
{
//获取v和w所在位置
int pv = GetVertexPos(g,v);
int pw = GetVertexPos(g,w);
if(pv==-1 || pw==-1)
return -1;
//从v的邻接顶点w的位置向后搜索,找到第一个与v相邻的顶点,即所求
for(int i=pw+1; i<g->NumVertices; ++i)
{
if(g->Edge[pv][i] == 1)
return i;
}
return -1;
}
销毁图
//销毁图
void DestroyGraph(GraphMtx *g)
{
//释放顶点
free(g->VerticesList);
g->VerticesList = NULL;
//释放边存储结构的列
for(int i=0; i<g->NumVertices; ++i)
{
free(g->Edge[i]);
}
free(g->Edge);//释放存放行指针的空间
g->Edge = NULL;
g->MaxVertices = g->NumEdges = g->NumVertices = 0;
}
结语
对图的邻接矩阵的介绍就到这里啦,希望这篇文章能给予你一些帮助,感谢各位人才的:点赞、收藏和评论,我们下次见。