C语言邻接表建立图详解

目录
  • 有向图
  • 无向图
  • 邻接表存图进行拓扑排序
  • 总结

有向图

代码:

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<stack>
using namespace std;
#define maxn 200
int v, e;
//表结点
typedef struct _Enode
{
	int ivex; //该边所指向的节点位置
	int value;//如果边有权值的话,就对其赋值
	struct _Enode* next_edge; //指向下一条边
}ENode,*PENode;
//头结点
typedef struct _VNode
{
	int data;
	ENode* fidt_edge;
}VNode;

//邻接表
typedef struct _LGraph
{
	int vex_num; //点的数量
	int edg_num; //边的数量
	VNode vexs[maxn]; //一维数组存表头节点
}LGraph;

LGraph* create()
{
	LGraph* pG;
	pG = (LGraph*)malloc(sizeof(LGraph));
	memset(pG, 0, sizeof(LGraph));
	pG->vex_num = v;  //顶点数
	pG->edg_num = e; //边数
	for (int i = 0; i < v; ++i) //初始化定点表的指针域为空
		pG->vexs[i].fidt_edge = NULL;
	//建立链表
	for (int i = 0; i < e; ++i)
	{
		int v1, v2;
		scanf_s("%d%d", &v1, &v2);
		ENode* p1 = (ENode*)malloc(sizeof(ENode));  //为新建的边申请空间
		p1->ivex = v2;//该边指向的节点
		// 头插法建立
		p1->next_edge = pG->vexs[v1].fidt_edge;
		pG->vexs[v1].fidt_edge = p1;
	}
	return pG;
}
int main()
{
	while (~scanf_s("%d%d", &v, &e))
	{
		if (v == 0 && e == 0)
			break;
		LGraph* pG;
		pG = create();
	}
	return 0;
}

无向图

在代码的建立链表的地方变成

//建立链表
	for (int i = 0; i < e; ++i)
	{
		int v1, v2;
		scanf_s("%d%d", &v1, &v2);
		ENode* p1 = (ENode*)malloc(sizeof(ENode));  //为新建的边申请空间
		p1->ivex = v2;//该边指向的节点
		// 头插法建立
		p1->next_edge = pG->vexs[v1].fidt_edge;
		pG->vexs[v1].fidt_edge = p1;
		//另一条边
		ENode* p2 = (ENode*)malloc(sizeof(ENode));  //为新建的边申请空间
		p2->ivex = v1;//该边指向的节点
		// 头插法建立
		p2->next_edge = pG->vexs[v2].fidt_edge;
		pG->vexs[v2].fidt_edge = p2;
	}

邻接表存图进行拓扑排序

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<stack>
using namespace std;
#define maxn 200
int v, e;
//表结点
typedef struct _Enode
{
	int ivex; //该边所指向的节点位置
	struct _Enode* next_edge; //指向下一条边
}ENode,*PENode;
//头结点
typedef struct _VNode
{
	int data;
	int indegree;//记录定点的入度
	ENode* fidt_edge;
}VNode;

//邻接表
typedef struct _LGraph
{
	int vex_num; //点的数量
	int edg_num; //边的数量
	VNode vexs[maxn]; //一维数组存表头节点
}LGraph;

LGraph* create()
{
	LGraph* pG;
	pG = (LGraph*)malloc(sizeof(LGraph));
	memset(pG, 0, sizeof(LGraph));
	pG->vex_num = v;  //顶点数
	pG->edg_num = e; //边数
	for (int i = 0; i < v; ++i) //初始化定点表的指针域为空
		pG->vexs[i].fidt_edge = NULL;
	for (int i = 0; i < e; ++i)
	{
		int v1, v2;
		scanf_s("%d%d", &v1, &v2);
		ENode* p1 = (ENode*)malloc(sizeof(ENode));  //为新建的边申请空间
		p1->ivex = v2;//该边指向的节点
		// 头插法建立
		p1->next_edge = pG->vexs[v1].fidt_edge;
		pG->vexs[v1].fidt_edge = p1;
	}
	return pG;
}
void TopSort(LGraph* pG)
{
	stack<int>s;
	int count, k, i;
	ENode* p;
	for (int i = 0; i < v; ++i) //记录各个顶点的入度
	{
		//遍历整个邻接表,如果表结点的值为 i,则i对应的头结点的入度加1
		p = pG->vexs[i].fidt_edge; //获得其指向的第一条边
		while (p)
		{
			pG->vexs[p->ivex].indegree++; //该边表存的位置对应的头结点的入度数量加1
			p = p->next_edge;
		}
	}
	//将入度为0的压入栈中
	for (int i = 0; i < v; ++i)
		if (pG->vexs[i].indegree == 0)s.push(i);
	count = 0;//对输出的顶点计数
	while (!s.empty())
	{
		int k = s.top(); //取出
		s.pop();
		++count;
		//与k节点相邻的节点的入度减1
		for (p = pG->vexs[k].fidt_edge; p; p = p->next_edge)
		{
			int to;
			to = p->ivex;
			pG->vexs[to].indegree--;
			//减为0的话就压入栈中
			if (pG->vexs[to].indegree == 0)
				s.push(to);
		}
	}
	if (count < pG->vex_num)
		printf("NO\n");
	else
		printf("YES\n");
}
int main()
{
	while (~scanf_s("%d%d", &v, &e))
	{
		if (v == 0 && e == 0)
			break;
		LGraph* pG;
		pG = create();
		TopSort(pG);
	}
	return 0;
}

总结

本篇文章就到这里了,希望能给你带来帮助,也希望您能够多多关注我们的更多内容!

(0)

相关推荐

  • C++实现有向图邻接表的构建

    本文实例为大家分享了C++实现有向图邻接表的构建代码,供大家参考,具体内容如下 数据结构里面的一道基础题,分享下自己的写法,验证可跑. #include<iostream> #include<string> const int MAX = 20; using namespace std; struct ArcNode { //弧结点 int adjvex = -1; //所指顶点位置 ArcNode *nextarc = nullptr; //下一条狐指针 size_t info

  • C++数据结构之实现邻接表

    本文实例为大家分享了C++数据结构之实现邻接表的具体代码,供大家参考,具体内容如下 一.图的邻接表实现 1.实现了以顶点顺序表.边链表为存储结构的邻接表: 2.实现了图的创建(有向/无向/图/网).边的增删操作.深度优先递归/非递归遍历.广度优先遍历的算法: 3.采用顶点对象列表.边(弧)对象列表的方式,对图的创建进行初始化:引用 "ObjArrayList.h"头文件,头文件可参看之前博文"数据结构之顺序列表(支持对象元素)"代码: 4.深度优先遍历分别采用递归/

  • C++实现有向图的邻接表表示

    本文实例为大家分享了C++有向图的邻接表表示,供大家参考,具体内容如下 一.思路: 有向图的插入有向边.删除边.删除顶点和无向图的有区别.其他的和无向图的类似. 1.插入有向边<e1, e2> 只需要插入<e1, e2>边就行,不需要插入对称边<e2, e1> 2.删除边<e1,e2>: 只需要删除<e1, e2>边就行,不需要仔找对称边<e2, e1>进行删除. 3.删除顶点v: 首先,要在邻接表中删除以v为头的边<v, w&

  • C++实现邻接表顶点的删除

    本文实例为大家分享了C++实现邻接表顶点的删除代码,供大家参考,具体内容如下 这里的边是无向边 删除顶点v时,要找到顶点v的邻接顶点w,把w中指向v的边删除掉,再删除边(v,w).循环这个过程,直到把和顶点v有关的边都删除掉为止. 再接着需要删除顶点v. 不可以直接像数组那样直接把顶点v之后的顶点位置像前移动一位,因为这样其他顶点的位置将会发生变化,顶点边中的顶点位置将会出错. 边和顶点的定义如下: struct Edge{//边节点的定义 int dest;//边的另一顶点位置 E cost;

  • C++实现图的邻接矩阵表示

    本文实例为大家分享了C++实现图的邻接矩阵表示代码,供大家参考,具体内容如下 1.遇到的问题:教材中写着子类Graphmtx(我用GrapMatrix)继承基类Graph 但是我在子类GraphMatrix中使用父类Graph的保护成员属性:maxVertices 显示没有声明(如下图). 原来,c++中声明一个模板类及子类,在子类中如果需要访问父类的protected变量,需要使用父类的类作用域限定符,否则会报"identifier not found"错误.如果不是模板类,可以直接

  • C语言邻接表建立图详解

    目录 有向图 无向图 邻接表存图进行拓扑排序 总结 有向图 代码: #include<stdio.h> #include<stdlib.h> #include<string.h> #include<stack> using namespace std; #define maxn 200 int v, e; //表结点 typedef struct _Enode { int ivex; //该边所指向的节点位置 int value;//如果边有权值的话,就对其

  • C语言*与&在操作线性表的作用详解

    在数据结构线性表一章,对线性表有这些操作方法(Operation): /*Operation*/ Initlist(*L);/*初始化操作,建立一个空的线性表L*/ ListEmpty(L):/*判断线性表是否为空表,若线性表为空,返回值为true,否则返回false*/ ClearList(*L):/*将线性表清空*/ GetElem(L,i,*e):/*性表L中的第i个位置元素值返回给e*/ LocateElem(L,e):/*在线性表L中查找与给定值e相等的元素,如果查找成功,返回该元素在

  • C语言实现静态顺序表的实例详解

    C语言实现静态顺序表的实例详解 线性表 定义一张顺序表也就是在内存中开辟一段连续的存储空间,并给它一个名字进行标识.只有定义了一个顺序表,才能利用该顺序表存放数据元素,也才能对该顺序表进行各种操作. 接下来看看静态的顺序表,直接上代码: SeqList.h #define _CRT_SECURE_NO_WARNINGS 1 #ifndef __SEQLIST_H__ #define __SEQLIST_H__ #include <stdio.h> #include <stdlib.h&g

  • C语言实现线性表的基本操作详解

    目录 前言 一.实训名称 二.实训目的 三.实训要求 四.实现效果 五.顺序存储代码实现 六.链式存储代码实现 前言 这里使用的工具是DEV C++ 可以借鉴一下 一.实训名称 线性表的基本操作 二.实训目的 1.掌握线性表的基本概念 2.掌握线性表的存储结构(顺序存储与链式存储) 3.掌握线性表的基本操作 三.实训要求 1.线性表可以顺序表也可以用单链表实现,鼓励大家用两种方式实现. 2.创建线性表时,数据从键盘输入整形数据 3.线性表类型定义和或各种操作的实现,可以用教材给出的方法,也可以自

  • R语言利用ggplot2绘制QQ图和箱线图详解

    目录 绘制qq图 函数介绍 例子 绘制boxplot 函数介绍 例子 利用分位点绘制箱线图 将QQ图和箱线图进行融合 函数介绍 参数介绍 注意事项 例子 绘制qq图 在ggplot2中绘制qq图需要两步,geom_qq()将绘制样本分位点,geom_qq_line()将绘制标准正态线 函数介绍 geom_qq() geom_qq( mapping = NULL, data = NULL, geom = "point", position = "identity",

  • 数据结构TypeScript之邻接表实现示例详解

    目录 图的结构特点 图的分类 图的表示 面向对象方法封装邻接表 构造函数 增加顶点和边 删除顶点和边 图的遍历 颜色标记 广度优先搜索(队列) 深度优先搜索(栈) 图的结构特点 图由顶点和顶点之间的边构成,记为G(V, E).其中V是顶点集合,E是边集合.作为一种非线性的数据结构,顶点之间存在多对多关系. 图的分类 无向图:两个顶点之间有两条互相关联的边.A和B之间为双向互通. 有向图:两个顶点之间有一条或两条关联的边.从A到B或者从B到A,只能单向通过. 带权无向图:在无向图的基础上增加一个权

  • C语言数据结构之单向链表详解分析

    链表的概念:链表是一种动态存储分布的数据结构,由若干个同一结构类型的结点依次串连而成. 链表分为单向链表和双向链表. 链表变量一般用指针head表示,用来存放链表首结点的地址. 每个结点由数据部分和下一个结点的地址部分组成,即每个结点都指向下一个结点.最后一个结点称为表尾,其下一个结点的地址部分的值为NULL(表示为空地址). 特别注意:链表中的各个结点在内存中是可以不连续存放的,具体存放位置由系统分配. 例如:int *ptr ; 因此不可以用ptr++的方式来寻找下一个结点. 使用链表的优点

  • C语言实现队列的示例详解

    目录 前言 一. 什么是队列 二. 使用什么来实现栈 三. 队列的实现 3.1头文件 3.2 函数的实现 四.完整代码 前言 前一段时间,我们试着用C语言实现了数据结构中的顺序表,单链表,双向循环链表,栈.今天我们再用C语言来实现另一种特殊的线性结构:队列 一. 什么是队列 队列是一种特殊的线性表,特殊之处在于它只允许在表的前端(head)进行删除操作,而在表的后端(tail)进行插入操作,和栈一样,队列是一种操作受限制的线性表.进行插入操作的端称为队尾,进行删除操作的端称为队头. 这个队列就可

  • C语言中的文件操作详解

    目录 1.为什么使用文件 2.什么是文件 2.1程序文件 2.2数据文件 2.3文件名 3.文件的打开和关闭 3.1文件指针 3.2文件的打开和关闭 4.文件的顺序读写 5.文件的随机读写 5.1fseek 5.2ftell 5.3rewind 6.文本文件和二进制文件 7.文件读取结束的判定 7.1被错误使用的feof 8.文件缓冲区 结论 1.为什么使用文件 在学习结构体时,写了一个简易的通讯录的程序,当程序运行起来的时候,可以在通讯录中增加和删除数据,此时数据是存放在内存当中的,当程序退出

  • C语言直接选择排序算法详解

    目录 1. 直接选择排序介绍 1.1 定义 1.2 基本原理 1.3 时间复杂度 1.4 空间复杂度 1.5 优缺点 2. 代码实现 2.1 代码设计 2.2 代码实现 1. 直接选择排序介绍 1.1 定义 直接选择排序是指每次都从剩余数据中选出最大或者最小的,将其排在已经排好的有序表后面. 1.2 基本原理 每次从无序表中选择最小(或最大)元素,将其作为首元素,知道所有元素排完为止.将一个有n个元素的数组从小到大排序,第一次从R[0] ~ R[n-1]中选取最小值,与R[0]交换,第二次从R[

随机推荐