C语言邻接表建立图详解

 更新时间:2021年08月25日 11:38:59   作者:落春只在无意间  
这篇文章主要介绍了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;
}

总结

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

相关文章

  • C++ 中约瑟夫环替换计数器m(数组解决)

    C++ 中约瑟夫环替换计数器m(数组解决)

    这篇文章主要介绍了C++ 中约瑟夫环替换计数器m(数组解决)的相关资料,需要的朋友可以参考下
    2017-05-05
  • C++趣味算法之侦探推理

    C++趣味算法之侦探推理

    本文详细讲解了C++趣味算法之侦探推理,文中通过示例代码介绍的非常详细。对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-12-12
  • OpenCV实现简单套索工具

    OpenCV实现简单套索工具

    这篇文章主要为大家详细介绍了OpenCV实现简单套索工具,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-01-01
  • OpenCV实现抠图工具

    OpenCV实现抠图工具

    这篇文章主要为大家详细介绍了OpenCV实现抠图工具,文中示例代码介绍的非常详细,具有一定为大家详细的参考价值,感兴趣的小伙伴们可以参考一下
    2022-01-01
  • C语言版二值图像统计连通区域

    C语言版二值图像统计连通区域

    这篇文章主要为大家详细介绍了C语言版二值图像统计连通区域的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-01-01
  • C++读取配置文件的示例代码

    C++读取配置文件的示例代码

    这篇文章主要介绍了C++读取配置文件的示例代码,帮助大家更好的理解和学习C++开发,感兴趣的朋友可以了解下
    2020-08-08
  • 下标操作符重载模拟多维数组详解

    下标操作符重载模拟多维数组详解

    虽然不能直接实现一对下标操作符重载,但是我们可以间接模拟。思路是这样的,先通过单下标操作返回一个具有下标操作能力的左值,对左值进行下标操作,两个下标操作表达式联立就实现了双下标操作
    2013-09-09
  • C++中based for循环的实现

    C++中based for循环的实现

    C++中的范围for循环是一种简洁的遍历容器的方法,本文主要介绍了C++中based for循环的实现,具有一定的参考价值,感兴趣的可以了解一下
    2025-02-02
  • C++ 中

    C++ 中"priority_queue" 优先级队列实例详解

    这篇文章主要介绍了C++ 中"priority_queue" 优先级队列实例详解的相关资料,需要的朋友可以参考下
    2017-04-04
  • C++ 折叠参数包详解(悄然增强编程效率)

    C++ 折叠参数包详解(悄然增强编程效率)

    折叠参数就是一个参数包, 代表是多个未知,tuple元组就是一个折叠参数的使用,这篇文章主要介绍了C++ 折叠参数包悄然增强编程效率,需要的朋友可以参考下
    2023-05-05

最新评论