关于带权无向图的一些操作

题目:根据图来建立它的邻接矩阵,通过邻接矩阵转化为邻接表,对邻接表进行深度优先访问和广度优先访问,最后用邻接矩阵生成它的最小生成树;

1.输入一个带权无向图(如下面图1和图2)的顶点数、边数、各条边信息(两个顶点和权值),建立该图的邻接矩阵结构,输出该邻接矩阵。

​ 图1 图2

2.将上述无向图邻接矩阵转换为邻接表结构,输出该邻接表;

3.根据该邻接表对无向图进行深度优先遍历序列和广度优先遍历序列,并输出遍历结果;

4.用prim算法实现构造该带权无向图的最小生成树,并将该最小生成树的各条边信息输出。

一些注意

程序用文件读入,使用前应写好读入文件;

初始工作(所有的结构体定义和函数声明)

结构体定义

# include <stdio.h>
# include <stdlib.h>
# define MAX 20
#define MAXV 20 //最大顶点个数
#define VertexType int //存放顶点信息
#define QElemType int
typedef int VetexType;
int visited[MAX];
typedef struct ArcNode //弧结点类型定义
{
int adjvex; //该弧的弧头
int weight;
struct ArcNode *nextarc; //指向另一个弧结点的指针
}ArcNode;
typedef struct VNode //邻接表头结点类型定义
{
VetexType data; //顶点信息
int adv;
ArcNode *firstarc; //指向第一个依附于该顶点的弧的指针
} AdjList[MAX];
typedef struct //图定义
{
AdjList vertices; //邻接表定义
int vexnum, arcnum; //顶点数和弧数
}ALGraph; typedef struct
{
int adj;
int weight; //顶点关系(0或1)
// InfoType info; //该弧或边的相关信息
} AdjMatrix[MAXV][MAXV];
typedef struct //图的定义
{
AdjMatrix arcs; //邻接矩阵
int vexnum, arcnum; //顶点数,弧数
VertexType vexs[MAXV]; //存放顶点信息
} MGraph;
//队列的定义
typedef struct QNode{
QElemType data;
struct QNode *next;
}QNode,*QueuePtr;
typedef struct {
QueuePtr front;
QueuePtr rear;
}LinkQueue;
//辅助数组定义
struct mini{
int adjvex;
int lowcost;
}mini,closedge[MAXV];

函数声明

void creatMG(MGraph *G);
void outMG(MGraph *G);
void Transition(MGraph *G1,ALGraph *G2);
void outALGraph(ALGraph *G);
void DFSTraverse(ALGraph *G);
void BFSTraverse(ALGraph G);
void DFS(ALGraph *G,int v);
void visit(int v,ALGraph *G);
void InitQueue(LinkQueue *Q);
void EnQueue(LinkQueue *Q,QElemType e);
void DeQueue(LinkQueue *Q,int *e);
int QueueEmpty(LinkQueue Q);
void MiniSpanTree(MGraph G,int u);

具体的函数实现

//创造邻接矩阵
void creatMG(MGraph *G){
int i, j, k,n,w,e;
FILE *fp1;
fp1=fopen("text2.txt","r");//第一组数据打开text1, 第二组数据打开text2;
if(!fp1)
{
printf("can't open file\n");
return ;
}
// printf("输入顶点数和边数\n");
fscanf(fp1,"%d%d", &n, &e); //输入顶点数n,边数e
G->arcnum =e;G->vexnum=n;
for(i=1; i<=n; i++)
for(j=1; j<=n; j++)
{
G->arcs[i][j].adj=0;
G->arcs[i][j].weight=1000;//假设最大权值不会超过一千
}
for(k=1; k<=e; k++) //勾画各条边
{
// printf("输入边(vi, vj)的顶点号vi, vj以及权值w\n ");
fscanf(fp1,"%d%d%d", &i, &j,&w); //输入一条边的两个顶点编号i,j
G->arcs[i][j].adj=1;
G->arcs[j][i].adj=1;
G->arcs[i][j].weight=w;
G->arcs[j][i].weight=w;
} //若是网,可让G[i][j]=w,w也需提前输入
} //creat_Mg
void outMG(MGraph *G){//输出邻接矩阵
int i, j;
for(i=1; i<=G->vexnum; i++) //矩阵原样输出
{
printf("\n");
for(j=1; j<=G->vexnum; j++)
printf("%5d", G->arcs[i][j].adj);
}
//输出所存在的边
for(i=1; i<=G->vexnum; i++)
{
for(j=1; j<=G->vexnum; j++)
{
if(G->arcs[i][j].adj!=0)
// printf("\n存在边( %d, %d )",i,j);
printf("\n存在边( %d, %d )它的权值为%d",i,j,G->arcs[i][j].weight);
}
}
printf("\n");
} //out_Mg
//邻接矩阵转化成邻接表
void Transition(MGraph *G1,ALGraph *G2){
int i,j;
ArcNode *p;
G2->arcnum=G1->arcnum;//点和边的数值相同
G2->vexnum=G1->vexnum;
for(i=1;i<=G2->vexnum;i++){
G2->vertices[i].data=i;
G2->vertices[i].adv=i;
G2->vertices[i].firstarc =NULL;//每个结点是首地址指向NULL
for(j=1;j<=G2->vexnum;j++){
if(G1->arcs[i][j].adj){
p=(ArcNode *)malloc(sizeof(ArcNode));
p->adjvex=j;
p->weight=G1->arcs[i][j].weight;//保存权重
p->nextarc=G2->vertices[i].firstarc;//头插入到后面
G2->vertices[i].firstarc =p;
}
}
} }
void outALGraph(ALGraph *G){ //输出邻接表
int i;
ArcNode *p;
for(i=1;i<=G->vexnum;i++){
printf("%d",G->vertices[i].adv);
p=G->vertices[i].firstarc;
while(p){//依次输出与该点相互邻接的点
printf("->%d",p->adjvex);
p=p->nextarc;
}
printf("\n");
}
}
//DFS
void DFSTraverse(ALGraph *G){
int i;
for(i=0;i<=G->vexnum;i++){
visited[i]=0;
}
for(i=1;i<=G->vexnum;i++){//防止不是连通图
if(!visited[i]){
DFS(G,i);
}
}
}
void DFS(ALGraph *G,int v){//深度优先遍历
ArcNode *p;
visited[v]=1;
visit(v,G);
for(p=G->vertices[v].firstarc;p;p=p->nextarc){//依次经过每个结点
if(!visited[p->adjvex]) DFS(G,p->adjvex);//未访问测访问
}
}
void BFSTraverse(ALGraph G) { //广度优先遍历
LinkQueue Q;
ArcNode *p;
int v,u;
for(v=0;v<=G.vexnum;v++){
visited[v]=0;//初始化
}
InitQueue(&Q);
for(v=1;v<=G.vexnum;v++){
if(!visited[v]){
visited[v]=1;
visit(v,&G);
EnQueue(&Q,v);
while(QueueEmpty(Q)){
DeQueue(&Q,&u);
for(p=G.vertices[u].firstarc;p;p=p->nextarc){
if(!visited[p->adjvex]){
visited[p->adjvex]=1;//新访问的点标记
visit(p->adjvex,&G);//访问
EnQueue(&Q,p->adjvex);//新点入队
}
}
}
}
}
}
void visit(int v,ALGraph *G){//访问点信息的函数
printf("%d\n",G->vertices[v].data);
}
void InitQueue(LinkQueue *Q){//队列初始化
Q->front=Q->rear=(QueuePtr)malloc(sizeof(QNode));
Q->front->next=NULL;//首尾都指向NULL
}
void EnQueue(LinkQueue *Q,int e){//入队
QueuePtr p;
p=(QueuePtr)malloc(sizeof(QNode));
p->data=e;
p->next=Q->rear->next;
Q->rear->next=p;//插入队中
Q->rear=p;
}
void DeQueue(LinkQueue *Q,int *e){//出队
QueuePtr p;
if(Q->front==Q->rear) return ;//队列为空
p=Q->front->next;
*e=p->data;
Q->front->next=p->next;
if(Q->rear==p) Q->rear=Q->front;//此时只有一个元素
}
int QueueEmpty(LinkQueue Q){
if(Q.front==Q.rear) return 0;//为空
return 1;
}
//构造最小生成树
void MiniSpanTree(MGraph G,int u){//u为顶点,表示从第几个顶点开始构造最小生成树
int j,i,k,a,min=1000;
k=u;
for(j=0;j<=G.vexnum;j++){
closedge[j].lowcost=1000;
}
for(j=1;j<G.vexnum;j++){
if(j!=k){
closedge[j].adjvex=u;
closedge[j].lowcost=G.arcs[k][j].weight;
}
}
closedge[k].lowcost=0;
for(i=2;i<=G.vexnum;i++){
for(a=1;a<=G.vexnum;a++){
if(closedge[a].lowcost<min&&closedge[a].lowcost!=0){
min=closedge[a].lowcost;
k=a;
}
}
min=1000;
printf("点%d到点%d,权值为%d\n",closedge[k].adjvex,k,closedge[k].lowcost);
closedge[k].lowcost=0;
for(j=1;j<=G.vexnum;j++){
if(G.arcs[k][j].weight<closedge[j].lowcost){//printf("-");
closedge[j].adjvex=k;
closedge[j].lowcost=G.arcs[k][j].weight;
}
}
}
}

main函数

//主函数
int main(){
MGraph G1;
ALGraph G2;
creatMG(&G1);
printf("输出邻接矩阵\n");
outMG(&G1);
Transition(&G1,&G2);
printf("输出邻接表\n");
outALGraph(&G2);
printf("输出深度优先排序\n");
DFSTraverse(&G2);
printf("输出广度优先排序\n");
BFSTraverse(G2);
printf("输出最小生成树\n");
MiniSpanTree(G1,1);
return 0;
}

一下是测试用例以及结果

第一组数据:

图的样子

邻接矩阵

各个点之间的权值

邻接表

DFS和BFS

最小生成树

图的建立以及应用(BFS,DFS,Prim)的更多相关文章

  1. 图的创建和遍历(BFS/DFS)

    图的表示方法主要有邻接矩阵和邻接表.其中邻接表最为常用,因此这里便以邻接表为例介绍一下图的创建及遍历方法. 创建图用到的结构有两种:顶点及弧 struct ArcNode { int vertexIn ...

  2. 邻结矩阵的建立和 BFS,DFS;;

    邻结矩阵比较简单,, 它的BFS,DFS, 两种遍历也比较简单,一个用队列, 一个用数组即可!!!但是邻接矩阵极其浪费空间,尤其是当它是一个稀疏矩阵的时候!!!-------------------- ...

  3. 数据结构代码整理(线性表,栈,队列,串,二叉树,图的建立和遍历stl,最小生成树prim算法)。。持续更新中。。。

    //归并排序递归方法实现 #include <iostream> #include <cstdio> using namespace std; #define maxn 100 ...

  4. 数据结构学习笔记05图 (邻接矩阵 邻接表-->BFS DFS、最短路径)

    数据结构之图 图(Graph) 包含 一组顶点:通常用V (Vertex) 表示顶点集合 一组边:通常用E (Edge) 表示边的集合 边是顶点对:(v, w) ∈E ,其中v, w ∈ V 有向边& ...

  5. 领接表的建立和它的DFS, BFS;;;

    //图的建立的实现->邻结矩阵和邻结表两种表示方法 #include <cstdio> #include <cstdlib> //#define _OJ_ int vis ...

  6. 图的建立(邻接矩阵)+深度优先遍历+广度优先遍历+Prim算法构造最小生成树(Java语言描述)

    主要参考资料:数据结构(C语言版)严蔚敏   ,http://blog.chinaunix.net/uid-25324849-id-2182922.html   代码测试通过. package 图的建 ...

  7. 【算法导论】图的深度优先搜索遍历(DFS)

    关于图的存储在上一篇文章中已经讲述,在这里不在赘述.下面我们介绍图的深度优先搜索遍历(DFS). 深度优先搜索遍历实在访问了顶点vi后,访问vi的一个邻接点vj:访问vj之后,又访问vj的一个邻接点, ...

  8. (二叉树 BFS DFS) leetcode 111. Minimum Depth of Binary Tree

    Given a binary tree, find its minimum depth. The minimum depth is the number of nodes along the shor ...

  9. Collect More Jewels(hdu1044)(BFS+DFS)

    Collect More Jewels Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Othe ...

随机推荐

  1. Postman设置自动捕获传递Cookie教程

    目录 前言 一.安装 1.Postman安装Install Interceptor Bridge 2.谷歌浏览器安装扩展Postman Interceptor 二.使用 1. 打开Capture Co ...

  2. 教你怎么设置Vegas渲染输出的选定范围

    在制作视频时,很多用户进行到渲染时,常常会发生这样那样的问题,导致导出的视频效果不甚理想.归结原因,还是用户在渲染输出时的选定范围存在问题. 接下来小编就为大家具体介绍下:vegas如何设置渲染输出的 ...

  3. 企业级LINUX自动化运维工具Ansible实战课程下载

    什么是Ansible? Ansible是新出现的自动化运维工具,基于Python开发,集合了众多运维工具(puppet.chef.func.fabric)的优点,实现了批量系统配置.批量程序部署.批量 ...

  4. 《Spring Boot 实战纪实》之如何攥写需求文档

    目录 前言 (思维篇)人人都是产品经理 1.需求文档 1.1 需求管理 1.2 如何攥写需求文档 1.3 需求关键点文档 2 原型设计 2.1 缺失的逻辑 2.2 让想法跃然纸上 3 开发设计文档 3 ...

  5. 虚拟机VMware15 Ubuntu18.04 搭建FTP服务器

    1.安装vsftpd sudo apt install vsftpd 2.查看是否安装成功,出现版本等信息即成功 sudo vsftpd -v 3.添加ftp用户 sudo useradd -m su ...

  6. 关于你天天见到的JDK、JRE和JVM

    什么是JDK.JRE.JVM? 大家都知道电脑的操作系统是由汇编和C语言写出,因此操作系统无法直接识别其他语言.这时我们就需要为我们写的Java程序配备一名翻译官 ----- 编译环境,将Java程序 ...

  7. 什么,kafka能够从follower副本读数据了 —kafka新功能介绍

    最近看了kafka2.4新版本的一些功能特性,不得不说,在kafka2.0以后,kafka自身就比较少推出一些新的feature了,基本都是一些修修补补的东西.倒是kafka connect和kafk ...

  8. 冲刺随笔——Day_Four

    这个作业属于哪个课程 软件工程 (福州大学至诚学院 - 计算机工程系) 这个作业要求在哪里 团队作业第五次--Alpha冲刺 这个作业的目标 团队进行Alpha冲刺 作业正文 正文 其他参考文献 无 ...

  9. 简单RTSCamera实现

    using System.Collections; using System.Collections.Generic; using UnityEngine; public class TopCamer ...

  10. 第15.23节 PyQt(Python+Qt)入门学习:Model/View架构中QListView视图配套Model的开发使用

    老猿Python博文目录 专栏:使用PyQt开发图形界面Python应用 老猿Python博客地址 一.概述 QListView理论上可以和所有QAbstractItemModel派生的类如QStri ...