C语言数据结构之图的遍历

输入一组顶点,建立无向图的邻接矩阵。输入一组顶点,建立有向图的邻接表。分别对无向图和有向图进行DFS(深度优先遍历)和BFS(广度优先遍历)。写出深度优先遍历的递归和非递归算法。根据建立的有向图,判断该图是否是有向无环图,若是,则输出其一种拓扑有序序列。

#include <stdio.h>
#include <stdlib.h>
#define MAX 20

typedef struct ArcNode{
	int adjvex;
	struct ArcNode *nextarc;
}ArcNode;

typedef struct{
	char data;
	ArcNode *firstarc;
}AdjList[MAX];

typedef struct{
	AdjList vertices;
	int vexnum;
	int arcnum;
}ALGraph;

typedef struct{
	int *base;
	int front,rear;
}CqQueue;

void InitQueue(CqQueue &Q)
{//初始化一个队列
	Q.base=(int*)malloc(MAX*sizeof(int));
	Q.front=Q.rear=0;
}

int QueueEmpty(CqQueue Q)
{//判断队列是否为空
	if(Q.rear==Q.front)
		return 1;
	return 0;
}

void EnQueue(CqQueue &Q,int e)
{//入队操作
	if((Q.rear+1)%MAX==Q.front)
		return;
	Q.base[Q.rear]=e;
	Q.rear=(Q.rear+1)%MAX;
}

void DeQueue(CqQueue &Q,int &e)
{//出队操作
	if(Q.rear==Q.front)
		return;
	e=Q.base[Q.front];
	Q.front=(Q.front+1)%MAX;
}

int LocateVex(ALGraph G,char v)
{//查找顶点v在图G中的位置
	for(int i=0;i<G.vexnum;i++)
		if(G.vertices[i].data==v)
			return i;
	return -1;


	for(int i=0;i<G.vexnum;i++)
		if(G.vexs[i]==v)
			return i;
	return -1;
}

void CreateAdjList(ALGraph &G)
{//建立无向图的邻接表
	int v,i,j,k;
	char v1,v2;
	ArcNode *p,*s;
	printf("输入无向图的顶点数和边数:\n");
	scanf("%d%d",&G.vexnum,&G.arcnum);
	getchar();
	printf("输入图的顶点信息:\n");
	for(v=0;v<G.vexnum;v++){
		scanf("%c",&G.vertices[v].data);getchar();
		G.vertices[v].firstarc=NULL;
	}
	
	printf("输入无向图的边:\n");
	for(k=0;k<G.vexnum;k++){
		scanf("%c%c",&v1,&v2);
		getchar();
		i=LocateVex(G,v1);
		j=LocateVex(G,v2);
		s=(ArcNode*)malloc(sizeof(ArcNode));
		s->adjvex=j;
		s->nextarc=NULL;
		if(!G.vertices[i].firstarc)
			G.vertices[i].firstarc=s;
		else{
			p=G.vertices[i].firstarc;
			while(p->nextarc)
				p=p->nextarc;
			p->nextarc=s;
		}
		s=(ArcNode*)malloc(sizeof(ArcNode));
		s->adjvex=i;
		s->nextarc=NULL;
		if(!G.vertices[j].firstarc)
			G.vertices[j].firstarc=s;
		else{
			p=G.vertices[j].firstarc;
			while(p->nextarc)
				p=p->nextarc;
			p->nextarc=s;
		}
	}
}

int visited[MAX];

void DFS(ALGraph G,int v)
{//从顶点v开始对图G进行深度优先搜索
	ArcNode *p;
	printf("%3c",G.vertices[v].data);
	visited[v]=1;
	for(p=G.vertices[v].firstarc;p;p=p->nextarc)
		if(!visited[p->adjvex])
			DFS(G,p->adjvex);
}

void DFSTraverse(ALGraph G)
{//对用邻接表存储的无向图G进行深度优先遍历
	int v;
	for(v=0;v<G.vexnum;v++)
		visited[v]=0;
	for(v=0;v<G.vexnum;v++)
		if(!visited[v])
			DFS(G,v);
}

void BFSTraverse(ALGraph G)
{//对用邻接表存储的无向图G进行深度优先遍历
	int u,v;
	CqQueue Q;
	ArcNode *p;
	for(v=0;v<G.vexnum;v++)
		visited[v]=0;
	InitQueue(Q);
	for(v=0;v<G.vexnum;v++)
		if(!visited[v]){
			printf("%3c",G.vertices[v].data);
			visited[v]=1;
			EnQueue(Q,v);
			while(!QueueEmpty(Q)){
				DeQueue(Q,u);
				for(p=G.vertices[u].firstarc;p;p=p->nextarc)
					if(!visited[p->adjvex]){
						printf("%3c",G.vertices[p->adjvex].data);
						visited[p->adjvex]=1;
						EnQueue(Q,p->adjvex);
					}
			}
		}
}

int main(){
	ALGraph G;
	printf("建立无向图的邻接表:\n");
	CreateAdjList(G);
	printf("无向图的深度优先遍历序列如下:\n");
	DFSTraverse(G);
	printf("\n\n无向图的广度优先遍历序列如下:\n");
	BFSTraverse(G);
	printf("\n");
	return 0;
}


转载于:https://www.cnblogs.com/zhuhengjie/p/5966941.html

  • 0
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
创建:包括建立结点的函数CreatVex(Graph *G),以及GreatUDG(Graph *G) ,GreatUDN(Graph *G) ,GreatDG(Graph *G) GreatDN(Graph *G) 1提示用户输入的基本信息:顶点数,边数以及的基本类型; 2通过for循环语句提示用户输入顶点的值; 3Graph结构体类型包括:AdjList用来存储头结点的数组;int类型vexnum和arcnum,用来表示顶点数和边数的变量;int类型kind,用来存储的类型。边ArcNode结构包括:adjvex,ArcNode *nextarc,int info前者表示指向的结点的下标,后者表示指向结点的下一条边结点,最后的变量为边所带的权值信息; 4根据的类型决定是否要使用边中的info变量; 5提示用户按照正确的形式输入边的端点以及边上的权值信息; 遍历:包括DFSTraverse(Graph G,VertexType vex)以及DFS(Graph G,int v)两个主要的便历函数。前者第二个参数表示开始进行便历的顶点,后者的第二个参数表示对的下标为v的顶点访问。 1遍历前首先建立一个标志数组Visited[],长度为中结点的数目,用来表示是否访问过一结点,访问前全置为0; 2接收用户要求开始访问的顶点,通过函数Adjfound(Graph G,VertexType c)找到改点在的结点中的下标; 3若该下标对应的标志数组的值为0,访问该下标的firstArcNode结点,同时把该结点的在访问标志数组中的值设置为1;若该下标对应的标志数组的值为1,则进行第5步; 4继续进行操作2; 5在标志数组中查找仍为0的项,得到下标值再进行第1步操作;如果都访问过则遍历结束。 6退出程序。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值