hnust 1964: 邻接表表示法

hnust 1964: 邻接表表示法

题目描述
输入一个图,用邻接表存储,并实现一些操作。
拷贝下面的代码,按要求完成其中的FirstAdjVex,NextAdjVex,sort和CreateUDG操作,其他地方不得改动。

//图的邻接表存储表示#include <iostream>
#include <iomanip>
using namespace std;#define MVNum 100         //最大顶点数
#define OK 1typedef string VerTexType; //顶点信息
typedef int OtherInfo;    //和边相关的信息//- - - - -图的邻接表存储表示- - - - -
typedef struct ArcNode {                //边结点int adjvex;                          //该边所指向的顶点的位置struct ArcNode *nextarc;          //指向下一条边的指针OtherInfo info;                      //和边相关的信息
} ArcNode;typedef struct VNode {VerTexType data;                    //顶点信息ArcNode *firstarc;                //指向第一条依附该顶点的边的指针
} VNode, AdjList[MVNum];                //AdjList表示邻接表类型typedef struct {AdjList vertices;                  //邻接表int vexnum, arcnum;              //图的当前顶点数和边数
} Graph;//得到顶点i的数据
VerTexType Vertexdata(const Graph &g, int i)
{return g.vertices[i].data;
}int LocateVex(const Graph &g, VerTexType v)
{//确定点v在G中的位置for(int i = 0; i < g.vexnum; ++i)if(g.vertices[i].data == v)return i;return -1;
}//LocateVex//返回v的第一个邻接顶点。若顶点在G中没有邻接表顶点,则返回-1。
int FirstAdjVex(const Graph &g, int v)
{/****在此下面完成代码***************//***********************************/
}// 返回v的(相对于w的)下一个邻接顶点。
int NextAdjVex(const Graph &g, int v, int w)
{/****在此下面完成代码***************//***********************************/
}//对每个顶点的链表排序,按顶点编号从小到大排列
void sort(ArcNode *arclist)
{/****在此下面完成代码***************//***********************************/
}int CreateUDG(Graph &g)
{//采用邻接表表示法,创建无向图G/****在此下面完成代码***************//***********************************/for(i = 0; i < g.vexnum; ++i) {           sort(g.vertices[i].firstarc); 保证有序,不依赖输入次序}//forreturn OK;
}//CreateUDGvoid DestroyUDG(Graph &g)
{//you should do this
}int main()
{Graph g;CreateUDG(g);//输出各个顶点的邻接点for(int i = 0; i < g.vexnum; i++) {cout << Vertexdata(g, i) << ":";for(int w = FirstAdjVex(g, i); w >= 0; w = NextAdjVex(g, i, w)) {cout << ' ' << Vertexdata(g, w);}cout << endl;}DestroyUDG(g);return 0;
}//main
输入
输入的第一行是两个整数,分别是图的总顶点数n和总边数e
第二行是n个空格分开的字符串,是顶点的名字,依次对应编号0~n-1。
随后有e行,每行两个空格分开的顶点名字,表示一条边的两个顶点。
具体见样例。
输出
输出n行,每行是第i个顶点的邻接顶点(要求按序号从小到大排列)。
具体见样例。
样例输入 Copy
8 9
v1 v2 v3 v4 v5 v6 v7 v8
v1 v2
v1 v3
v2 v4
v2 v5
v3 v6
v3 v7
v4 v8
v5 v8
v6 v7
样例输出 Copy
v1: v2 v3
v2: v1 v4 v5
v3: v1 v6 v7
v4: v2 v8
v5: v2 v8
v6: v3 v7
v7: v3 v6
v8: v4 v5

提示
样例对应教材6.5的图G4

解题过程

邻接表表示法(链式)
顶点: 按编号顺序将顶点数据存储在一维数组中。
关联同一顶点的边: 用线性链表存储。
如果有边\弧的信息,还可以在表结点中增加一项,

无向图的邻接表
特点:
邻接表不唯一
若无向图中有n个顶点、e条边,则其邻接表需要n个头结点和2e个表结点。适宜存储稀疏图。
无向图中顶点v i 的度为第i个单链表中的结点数。

有向图的邻接表
邻接表特点:
顶点 v i 的出度为第i个单链表中的结点个数。
顶点v i 的入度为整个单链表中邻接点域值是i-1的结点个数。
找出度易,找入度难

逆邻接表特点:
顶点 v i 的入度为第i个单链表中的结点个数。
顶点v i 的出度为整个单链表中邻接点域值是i-1的结点个数。
找入度易,找出度难
当邻接表的存储结构形成后,图便唯一确定。

这段C++代码实现了无向图的邻接表表示、创建、遍历和销毁。以下是对代码的详细解析:

1. 头文件和命名空间

  • 包含<iostream><iomanip>,提供输入输出流和格式化功能。
  • 使用using namespace std;简化代码。

2. 宏定义和类型定义

  • MVNum定义了最大顶点数。
  • VerTexType定义顶点信息的数据类型为string
  • OtherInfo定义边相关信息的数据类型为int

3. 图的邻接表结构

  • ArcNode结构体定义了边结点,包含邻接顶点索引、指向下一条边的指针和边的相关信息。
  • VNode结构体定义了顶点,包含顶点信息和指向第一条依附该顶点的边的指针。
  • Graph结构体定义了图,包含邻接表数组和图的顶点数、边数。

4. 函数定义

  • Vertexdata:获取顶点的数据。
  • LocateVex:确定顶点在图中的位置,如果找到则返回索引,否则返回-1。
  • FirstAdjVex:返回顶点的第一个邻接顶点的索引,如果没有邻接点则返回-1。
  • NextAdjVex:返回顶点相对于给定邻接顶点的下一个邻接顶点的索引,如果没有则返回-1。
  • sort:对邻接链表进行排序,按顶点编号从小到大排列。

5. 图的创建函数CreateUDG

  • 读取顶点数和边数。
  • 读取顶点信息并初始化邻接表。
  • 循环读取边的信息,使用LocateVex找到顶点的索引,并创建边结点添加到邻接表中。
  • 调用sort函数对所有顶点的邻接链表进行排序。

6. 图的销毁函数DestroyUDG

  • 该函数需要实现释放图占用的内存资源,但当前代码为空。

7. 主函数main

  • 创建图g并调用CreateUDG函数。
  • 遍历所有顶点,输出每个顶点及其邻接点。
  • 调用DestroyUDG函数销毁图。

代码逻辑分析

  • 代码使用邻接表表示无向图,允许多个顶点共享相同的邻接链表。
  • 使用CreateUDG函数根据用户输入创建图,包括顶点和边的信息。
  • 使用FirstAdjVexNextAdjVex函数遍历每个顶点的邻接点。
  • 使用sort函数对邻接链表进行排序,以便于输出和后续操作。

潜在问题

  • DestroyUDG函数未实现,没有释放图占用的内存。
  • sort函数中的排序逻辑可能不正确,因为它只是简单地交换了邻接点的索引,而没有考虑链表的结构。

改进建议

  • 实现DestroyUDG函数,以正确释放图占用的内存资源。
  • 优化sort函数,确保邻接链表可以正确排序。
  • 考虑使用std::vector代替固定大小的数组,以提高代码的灵活性和安全性。
  • 对输入数据进行有效性检查,确保它们在预期的范围内,并且是有效的图数据。

AC代码

#include <iostream>
#include <iomanip>
using namespace std;#define MVNum 100         //最大顶点数
#define OK 1
typedef string VerTexType; //顶点信息
typedef int OtherInfo;    //和边相关的信息//- - - - -图的邻接表存储表示- - - - -
typedef struct ArcNode                  //边结点
{int adjvex;                          //该节点指向的对应顶点是谁,即回答自己是谁struct ArcNode *nextarc;          //指向下一条边的指针OtherInfo info;                      //和边相关的信息
} ArcNode;typedef struct VNode
{VerTexType data;                    //顶点信息ArcNode *firstarc;                //指向第一条依附该顶点的边的指针
} VNode, AdjList[MVNum];                //AdjList表示邻接表类型typedef struct
{AdjList vertices;                  //邻接表int vexnum, arcnum;              //图的当前顶点数和边数
} Graph;//得到顶点i的数据
VerTexType Vertexdata(const Graph &g, int i)
{return g.vertices[i].data;
}int LocateVex(const Graph &g, VerTexType v)
{//确定点v在G中的位置for(int i = 0; i < g.vexnum; ++i)if(g.vertices[i].data == v)return i;return -1;
}//LocateVex//返回v的第一个邻接顶点。若顶点在G中没有邻接表顶点,则返回-1。
int FirstAdjVex(const Graph &g, int v)
{/****在此下面完成代码***************/if(g.vertices[v].firstarc)return   g.vertices[v].firstarc->adjvex;return -1;/***********************************/
}// 返回v的(相对于w的)下一个邻接顶点。
int NextAdjVex(const Graph &g, int v, int w)
{/****在此下面完成代码***************/ArcNode *p1=g.vertices[v].firstarc;while(p1->adjvex!=w)p1=p1->nextarc;if(p1->nextarc)return p1->nextarc->adjvex;return -1;/***********************************/
}//对每个顶点的链表排序,按顶点编号从小到大排列
void sort(ArcNode *arclist)
{/****在此下面完成代码***************/ArcNode *p1,*p2;for(p1=arclist; p1; p1=p1->nextarc){for(p2=p1->nextarc; p2; p2=p2->nextarc){if(p1->adjvex>p2->adjvex)swap(p1->adjvex,p2->adjvex);}}/***********************************/
}int CreateUDG(Graph &g)
{//采用邻接表表示法,创建无向图G/****在此下面完成代码***************/cin>>g.vexnum>>g.arcnum;for(int i=0; i<g.vexnum; i++){cin>>g.vertices[i].data;g.vertices[i].firstarc=NULL;}while(g.arcnum--){string v1,v2;cin>>v1>>v2;int h1=LocateVex(g,v1),h2=LocateVex(g,v2);if(h1!=-1&&h2!=-1){ArcNode*p1=new ArcNode;p1->adjvex=h2;p1->nextarc=g.vertices[h1].firstarc;g.vertices[h1].firstarc=p1;ArcNode*p2=new ArcNode;p2->adjvex=h1;p2->nextarc=g.vertices[h2].firstarc;g.vertices[h2].firstarc=p2;}}/***********************************/for(int i = 0; i < g.vexnum; ++i){sort(g.vertices[i].firstarc); }//forreturn OK;
}//CreateUDGvoid DestroyUDG(Graph &g)
{//you should do this
}int main()
{Graph g;CreateUDG(g);//输出各个顶点的邻接点for(int i = 0; i < g.vexnum; i++){cout << Vertexdata(g, i) << ":";for(int w = FirstAdjVex(g, i); w >= 0; w = NextAdjVex(g, i, w)){cout << ' ' << Vertexdata(g, w);}cout << endl;}DestroyUDG(g);return 0;
}//main

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

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

相关文章

Java高级重点知识点-25-Stream流、方法引用

文章目录 Stream流流式思想概述获取流常用方法 方法引用方法引用符通过对象名引用成员方法通过类名称引用静态方法通过super引用成员方法通过this引用成员方法类的构造器引用数组的构造器引用 Stream流 通过循环遍历来讲解流的优势&#xff1b; 要求&#xff1a;筛选所有姓张的…

实现Android夜间模式主题:从入门到精通

实现Android夜间模式主题:从入门到精通 随着用户对夜间模式的需求越来越高,Android开发者需要掌握如何在应用中实现夜间模式。本文将详细介绍在Android中实现夜间模式的步骤,包括配置、实现、以及一些最佳实践,帮助开发者创建更具吸引力和用户友好的应用。 夜间模式的优势…

智能交互中意图的损失、补全与弥聚

智能交互中的“意图的损失”和“意图的补全”通常用于描述在交流过程中可能出现的信息不完整或不清晰导致的情况。意图的损失指的是在信息传递过程中&#xff0c;发送者的意图未能完全或准确地被接收者理解的情况。这种情况可能由多种因素导致&#xff0c;如信息不完整&#xf…

Redis基础教程(二十):Java使用Redis

&#x1f49d;&#x1f49d;&#x1f49d;首先&#xff0c;欢迎各位来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里不仅可以有所收获&#xff0c;同时也能感受到一份轻松欢乐的氛围&#xff0c;祝你生活愉快&#xff01; &#x1f49d;&#x1f49…

华贝甄选干细胞科技,揭秘生命修复的奥秘

在探索生命奥秘的漫漫征途中&#xff0c;华贝甄选凭借干细胞科技的神奇力量&#xff0c;为您点亮健康与活力的希望之光。 我们深知&#xff0c;细胞是生命的基石&#xff0c;而干细胞则是这基石中蕴含的无限潜能。华贝甄选精心打造的干细胞疗法&#xff0c;如同神奇的魔法&…

大模型融入云平台,信息化走向数智化

随着信息技术的飞速发展&#xff0c;我们已经见证了从数据化到信息化的转变。然而&#xff0c;随着人工智能技术的不断突破&#xff0c;我们迎来了一个全新的时代——数智化时代。在这个时代&#xff0c;大模型与云平台的融合成为了推动信息化向数智化转变的关键力量。 大型模型…

详细说一下vue中的路由拦截器的作用

在Vue中&#xff0c;路由拦截器主要用于在导航到某个路由前或者离开某个路由时进行拦截和处理。这种机制允许开发者在路由导航发生前或者发生后执行特定的逻辑&#xff0c;比如权限验证、数据加载、页面跳转等。 在Vue Router中&#xff0c;可以通过以下几种方式来实现路由拦截…

决策树算法介绍,原理与案例实现

决策树算法是一种非常受欢迎的机器学习算法&#xff0c;它能够用于分类和回归任务。以下是决策树算法的详细介绍&#xff0c;包括原理和案例实现&#xff0c;以及相应的Python代码。 决策树算法介绍 基本概念 决策树是一种树形结构&#xff0c;用于对数据进行分类或回归。它…

实现前端用户密码重置功能(有源码)

引言 密码重置功能是任何Web应用程序中至关重要的一部分。当用户忘记密码时&#xff0c;密码重置功能可以帮助他们安全地重设密码。本文将介绍如何使用HTML、CSS和JavaScript&#xff08;包括Vue.js&#xff09;来实现前端的密码重置功能。 1. 项目结构 首先&#xff0c;我们…

2024SpringCloud学习笔记

远程调用Rest Template 服务注册与发现&分布式配置管理 Consul 下载安装 官网https:/ldeveloper.hashicorp.com/consul/downloads 开发者模式启动consul agennt -dev 浏览器访问本地端口:8500 服务注册与发现 Maven引入 <!--SpringCloud consul discovery -->…

【Python实战因果推断】31_双重差分2

目录 Canonical Difference-in-Differences Diff-in-Diff with Outcome Growth Canonical Difference-in-Differences 差分法的基本思想是&#xff0c;通过使用受治疗单位的基线&#xff0c;但应用对照单位的结果&#xff08;增长&#xff09;演变&#xff0c;来估算缺失的潜…

小阿轩yx-NoSQL 之 Redis 配置与优化

小阿轩yx-NoSQL 之 Redis 配置与优化 Redis 数据库介绍 是一个非关系型数据库 关系数据库与非关系型数据库 按照数据库结构划分的 关系型数据库 是一个结构化的数据库&#xff0c;创建在关系模型基础上&#xff0c;一般面向于记录借助集合代数等数学概念和方法处理数据库…

215.Mit6.S081-实验三-page tables

在本实验室中&#xff0c;您将探索页表并对其进行修改&#xff0c;以简化将数据从用户空间复制到内核空间的函数。 一、实验准备 开始编码之前&#xff0c;请阅读xv6手册的第3章和相关文件&#xff1a; kernel/memlayout.h&#xff0c;它捕获了内存的布局。kernel/vm.c&…

Python:Python基础知识(注释、命名、数据类型、运算符)

.注释 Python有两种注释方法&#xff1a;单行注释和多行注释。单行注释以#开头&#xff0c;多行注释以三个单引号 或三个双引号 """ 开头和结尾。 2.命名规则 命名规则: 大小写字母、数字、下划线和汉字等字符及组合&#xff1b; 注意事项: 大小写敏感、首…

Linux环境下Oracle 11g的离线安装与配置历程

在成功体验了 Windows 版本的Oracle 11g 后&#xff0c;这几天心血来潮&#xff0c;决定再挑战一下Linux 环境下的安装&#xff0c;特别是在考虑到部门内部虚拟机无法联网的情况下&#xff0c;我选择了在CentOS 7上进行离线安装。这次安装之旅&#xff0c;主要参考了下面大佬的…

【深度学习】关于模型加速

模型转为半精度的会加快推理速度吗 将模型转为半精度&#xff08;通常指16位浮点数&#xff0c;即FP16&#xff09;确实可以加快推理速度&#xff0c;同时还能减少显存&#xff08;GPU内存&#xff09;的使用。以下是一些关键点&#xff1a; 加快推理速度的原因 减少计算量&a…

【计算机科学】CCF-C特刊征稿合集,见刊快,期刊质量高,速投!

期刊推荐 期刊名称&#xff1a;ACTA INFORMATICA 主题包括以下项目的理论方面。 算法及其分析 自动机和形式语言 可计算性和复杂性 数据处理 离散数学 逻辑学&#xff08;计算机科学&#xff09; 人工智能的数学基础 编程语言理论 安全 系统理论 验证 中科院四区 …

不会电脑编程怎么编程:零基础入门指南

不会电脑编程怎么编程&#xff1a;零基础入门指南 在这个数字化时代&#xff0c;编程技能已经成为了一种重要的竞争力。然而&#xff0c;对于许多没有接触过编程的人来说&#xff0c;如何入门却成为了一个令人困惑的问题。如果你也不会电脑编程&#xff0c;那么本文将为你提供…

STM32智能物流机器人系统教程

目录 引言环境准备智能物流机器人系统基础代码实现&#xff1a;实现智能物流机器人系统 4.1 数据采集模块 4.2 数据处理与导航算法 4.3 通信与网络系统实现 4.4 用户界面与数据可视化应用场景&#xff1a;物流机器人管理与优化问题解决方案与优化收尾与总结 1. 引言 智能物流…

synchronized (userAccount.intern())知识点

synchronized (userAccount.intern()) 是一种在 Java 中同步代码块的方法&#xff0c;用于确保在多线程环境中对共享资源的安全访问。具体来说&#xff0c;这个语句使用 userAccount.intern() 返回的对象作为锁来同步代码块&#xff0c;以确保同一时刻只有一个线程能够执行该代…