前往小程序,Get更优阅读体验!
立即前往
发布
社区首页 >专栏 >【数据结构——图】图的邻接矩阵和邻接表的存储(头歌实践教学平台习题)【合集】

【数据结构——图】图的邻接矩阵和邻接表的存储(头歌实践教学平台习题)【合集】

作者头像
Rossy Yan
修改2024-12-26 10:29:47
修改2024-12-26 10:29:47
12200
代码可运行
举报
运行总次数:0
代码可运行

任务描述

本关任务:编写一个程序实现图的邻接矩阵和邻接表的存储。

相关知识

为了完成本关任务,你需要掌握:

  1. 带权有向图
  2. 图的邻接矩阵,
  3. 图的邻接表
1. 带权有向图

针对有向图的邻接矩阵和邻接表的存储,如下列图形:

2. 图的邻接矩阵

若 G 为带权有向图,其邻接矩阵 A 中的元素 Aij 遵循以下规则进行赋值:

  1. 当 i≠j,并且存在从顶点 i 指向顶点 j 的有向边,即 <i, j>∈E (G) 时,此时 Aij 的值等于该有向边的权值 wij;
  2. 当 i = j 时,Aij 的取值为 0,这表示顶点到自身的权值为 0;
  3. 而在其他情况下,也就是不存在从顶点 i 指向顶点 j 的有向边时,Aij 的值设定为无穷大(通常用特定的极大数值来表示,例如在程序实现中可以采用类似 INT_MAX 这样能表示极大值的常量来表示无穷大的概念)。

示例代码如下:

代码语言:javascript
代码运行次数:0
复制
 if (g.edges[i][j]!=INF)
     printf("%d ",g.edges[i][j]);
 else
     printf("%s ","∞");

(INF表示无穷大,表示整数:32767)

3. 图的邻接表

邻接表结点由三个域组成:

  1. adjvex指示与顶点vi邻接的点在图中的位置,
  2. nextarc指示下一条边或弧的结点,
  3. info存储与边或弧的权值。

表头结点由两个域组成:

  1. data存储顶点vi的名称或其他信息,
  2. firstarc指向链表中第一个结点。

示例代码如下:

代码语言:javascript
代码运行次数:0
复制
     for (int i=0;i<G->n;i++)
     {
         p=G->adjlist[i].firstarc;
         printf("%3d: ",i);
         while (p!=NULL)
         {
             printf("%3d[%d]→",p->adjvex,p->weight);
             p=p->nextarc;
         }
         printf("∧\n");
     }

测试说明

平台会对你编写的代码进行测试:

测试输入:( 输入图的顶点数和边数,再输入图的邻接矩阵。)

代码语言:javascript
代码运行次数:0
复制
6 10 0 5 32767 7 32767 32767 32767 0 4 32767 32767 32767 8 32767 0 32767 32767 9 32767 32767 5 0 32767 6 32767 32767 32767 5 0 32767 3 32767 32767 32767 1 0

预期输出:(Prim算法求解结果)

代码语言:javascript
代码运行次数:0
复制
边(0,5)权为:3

边(5,4)权为:1

边(0,1)权为:5 

边(1,2)权为:4

边(4,3)权为:5

开始你的任务吧,祝你成功!


通关代码

代码语言:javascript
代码运行次数:0
复制
#include <algorithm>
#include <iostream>
#include <limits>
#include <vector>
using namespace std;
const int MAX_VERTICES = 6; // 最大顶点数
const int INF = 32767;      // 表示无穷大

// 邻接矩阵表示的图
struct GraphMatrix {
  int n;                     // 顶点数
  vector<vector<int>> edges; // 邻接矩阵
};

// 邻接表的边节点
struct ArcNode {
  int adjvex;       // 邻接点在图中的位置
  int weight;       // 边的权重
  ArcNode *nextarc; // 指向下一条边
};

// 邻接表
struct VNode {
  int data;          // 顶点信息
  ArcNode *firstarc; // 指向第一条边
};

struct GraphList {
  int n;                 // 顶点数
  vector<VNode> adjlist; // 邻接表
};

// 输入邻接矩阵并构建图
void inputGraphMatrix(GraphMatrix &g) {
  cin >> g.n; // 读入顶点数
  int e;      // 边数
  cin >> e;   // 读入边数
  g.edges.resize(g.n, vector<int>(g.n, INF));

  for (int i = 0; i < g.n; ++i) {
    g.edges[i][i] = 0; // 对角线元素为0
  }

  // 读入邻接矩阵
  for (int i = 0; i < g.n; ++i) {
    for (int j = 0; j < g.n; ++j) {
      cin >> g.edges[i][j];
    }
  }
}

// 输出邻接矩阵
void printGraphMatrix(const GraphMatrix &g) {
  cout << "(1)图G的邻接矩阵:" << endl;
  for (int i = 0; i < g.n; ++i) {
    for (int j = 0; j < g.n; ++j) {
      if (g.edges[i][j] == INF) {
        cout << "∞ ";
      } else {
        cout << g.edges[i][j] << " ";
      }
    }
    cout << endl;
  }
}

// 构建邻接表
void buildGraphList(const GraphMatrix &gm, GraphList &gl) {
  gl.n = gm.n;
  gl.adjlist.resize(gl.n);

  for (int i = 0; i < gm.n; ++i) {
    gl.adjlist[i].data = i;           // 存储顶点
    gl.adjlist[i].firstarc = nullptr; // 初始化第一条边
  }

  // 从邻接矩阵构建邻接表
  for (int i = 0; i < gm.n; ++i) {
    for (int j = 0; j < gm.n; ++j) {
      if (gm.edges[i][j] != INF && gm.edges[i][j] != 0) {
        ArcNode *arc = new ArcNode(); // 动态分配新边
        arc->adjvex = j;              // 指向邻接点
        arc->weight = gm.edges[i][j]; // 边的权重
        arc->nextarc = nullptr;       // 初始化下一条边为nullptr

        // 将新节点插入已排序的链表中
        if (gl.adjlist[i].firstarc == nullptr ||
            gl.adjlist[i].firstarc->adjvex > j) {
          arc->nextarc = gl.adjlist[i].firstarc; // 插入到链表头
          gl.adjlist[i].firstarc = arc;
        } else {
          ArcNode *p = gl.adjlist[i].firstarc;
          while (p->nextarc != nullptr && p->nextarc->adjvex < j) {
            p = p->nextarc; // 寻找插入位置
          }
          arc->nextarc = p->nextarc; // 插入
          p->nextarc = arc;
        }
      }
    }
  }
}

// 输出邻接表
void printGraphList(const GraphList &gl) {
  cout << "(2)图G的邻接表:" << endl;
  for (int i = 0; i < gl.n; ++i) {
    cout << "  " << i << ": ";
    ArcNode *p = gl.adjlist[i].firstarc;
    while (p != nullptr) {
      cout << "  " << p->adjvex << "[" << p->weight << "]→";
      p = p->nextarc;
    }
    cout << "∧" << endl;
  }
}

int main() {
  GraphMatrix gm;
  inputGraphMatrix(gm);
  printGraphMatrix(gm);

  GraphList gl;
  buildGraphList(gm, gl);
  printGraphList(gl);

  return 0;
}

测试结果

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2024-12-24,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 任务描述
  • 相关知识
    • 1. 带权有向图
    • 2. 图的邻接矩阵
    • 3. 图的邻接表
  • 测试说明
  • 通关代码
  • 测试结果
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档