1、图的定义

图:是一种灵活的数据结构。一般作为一种模型用来定义对象之间的关系或者联系。

对象由顶点表示,而对象之间的关系或关联则通过顶点之间的边来表示。

2、图的应用

图算法、统计网络跳数、拓扑排序、图着色、哈密顿圈问题、分团问题、可序列化冲突

3、图的代码实现

/*graph.h*/
#ifndef GRAPH_H
#define GRAPH_H
#include <stdlib.h>
#include "list.h"
#include "set.h"

/*define a structure for adjacency lists*/
typedef struct AdjList_
{
void *vertex;
Set adjacent;
}AdjList;

/*define structure for graphs*/
typedef struct Graph_
{
int vcount;
int ecount;
int (*match)(const void *key1,const void *key2);
void (*destroy)(void *data);
List adjlists;
}Graph;

/*define colors for vertices in graphs*/
typedef enum VertexColor_ {white,gray,black} VertexColor;

/*public interface*/
void graph_init(Graph *graph,int (*match)(const void *key1,const void *key2),void (*destroy)(void *data));
void graph_destroy(Graph *graph);
int graph_ins_vertex(Graph *graph,const void *data);
int graph_ins_edge(Graph *graph,const void *data1,const void *data2);
int graph_rem_vertex(Graph *graph,void **data);
int graph_rem_edge(Graph *graph,void *data1,void **data2);
int graph_adjlist(const Graph *graph,const void *data,AdjList **adjlist);
int graph_is_adjacent(const Graph *graph,const void *data1,const void *data2);
#define graph_adjlists(graph) ((graph)->dajlists)
#define graph_vcount(graph) ((graph)->vcount)
#define graph_ecount(graph) ((graph)->ecount)

#endif


 

/*graph.c*/
#include<stdlib.h>
#include<string.h>
#include"graph.h"
#include"list.h"
#include"set.h"

void graph_init(Graph *graph,int (*match)(const void *key1,const void *key2),void (*destroy)(void *data))
{
/*initialize the graph*/
graph->vcount=0;
graph->ecount=0;
graph->match=match;
graph->destroy=destroy;

/*initialize the list of adjacentcy-list structure*/
list_init(&graph->adjlists,NULL);
return ;
}


void graph_destroy(Graph *graph)
{
AdjList *adjlist;

/*remove each adjacency-list structure and destroy its adjacency list*/
while(list_size(&graph->adjlists)>0)
{
if(list_rem_next(&graph->adjlists,NULL,(void **)&adjlist)==0)
{
set_destroy(&adjlist->adjacent);
if(graph->destroy!=NULL)
graph->destroy(adjlist->vertex);
free(adjlist);
}
}

/*destroy the list of adjacency-list structures,which is now empty*/
list_destroy(&graph->adjlists);

/*clear the structure*/
memset(graph,0,sizeof(Graph));
return ;
}


int graph_ins_vertex(Graph *graph,const void *data)
{
ListElmt *element;
AdjList *adjlist;
int retval;

/*do not allow the insertion of duplicate vertices*/
for(element=list_head(&graph->adjlists);element !=NULL;element=list_next(element))
{
if(graph->match(data,((AdjList *)list_data(element))->vertex))
return 1;
}

/*insert the vertex*/
if((adjlist=(AdjList *)malloc(sizeof(AdjList)))==NULL)
return -1;
adjlist->vertex=(void *)data;
set_init(&adjlist->adjacent,graph->match,NULL);
if((retval=list_ins_next(&graph->adjlists,list_tail(&graph->adjlists),adjlist))!=0)
{
return retval;
}

/*adjust the vertex count to account for the inserted vetex*/
graph->vcount++;
return 0;
}


int graph_ins_edge(Graph *graph,const void *data1,const void *data2)
{
ListElmt *element;
int retval;

/*do not allow insertion of an edge without both its vertices in the graph*/
for(element=list_head(&graph->adjlists);element !=NULL;element=list_next(element))
{
if(graph->match(data2,((AdjList *)list_data(element))->vertex))
break;
}
if(element==NULL)
return -1;
for(element=list_head(&graph->adjlists);element !=NULL;element=list_next(element))
{
if(graph->match(data1,((AdjList *)list_data(element))->vertex))
break;
}
if(element==NULL)
return -1;

/*insert the second vertex into the adjacency list of the first vertex*/
if((retval=set_insert(&((AdjList *)list_data(element))->adjacent,data2))!=0)
{
return retval;
}

/*Adjust the edge count to account for the inserted edge*/
graph->ecount++;
return 0;
}


int graph_rem_vertex(Graph *graph,void **data)
{
ListElmt *element,*temp,*prev;
AdjList *adjlist;
int found;

/*traverse each adjacency list and the vertice it contains*/
prev=NULL;
found=0;
for(element=list_head(&graph->adjlists);element !=NULL;element=list_next(element))
{
/*do not allow removal of the vertex if it is in an adjacency list*/
if(set_is_member(&((AdjList *)list_data(element))->adjacent,*data))
return -1;

/*keep a pointer to the vertex to be removed*/
if(graph->match(*data,((AdjList *)list_data(element))->vertex))
{
temp=element;
found=1;
}

/*keep a pointer to the vertex before the vertex to be removed*/
if(!found)
prev=element;
}

/*return if the vertex was not found*/
if(!found)
return -1;

/*do not allow removal of the tex if its adjacency list is not empty*/
if(set_size(&((AdjList *)list_data(temp))->adjacent)>0)
return -1;

/*remove the vertex*/
if(list_rem_next(&graph->adjlists,prev,(void **)&adjlist)!=0)
return -1;

/*free the storage allocated by the abstract datatype*/
*data=adjlist->vertex;
free(adjlist);

/*adjust the vertex count to account for the removed vertex*/
graph->vcount--;
return 0;

}


int graph_rem_edge(Graph *graph,void *data1,void **data2)
{
ListElmt *element;

/*locate the adjacency list for the first vertex*/
for(element=list_head(&graph->adjlists);element !=NULL;element=list_next(element))
{
if(graph->match(data1,((AdjList *)list_data(element))->vertex))
break;
}
if(element==NULL)
return -1;

/*remove the second vertex from the adjacency list of the first vertex*/
if(set_remove(&((AdjList *)list_data(element))->adjacent,data2)!=0)
return -1;

/*adjust the edge count to account for the removed edge*/
graph->ecount--;
return 0;
}


int graph_adjlist(const Graph *graph,const void *data,AdjList **adjlist)
{
ListElmt *element,*prev;

/*locate the adjacency list for the vertex*/
prev=NULL;
for(element=list_head(&graph->adjlists);element !=NULL;element=list_next(element))
{
if(graph->match(data,((AdjList *)list_data(element))->vertex))
break;
}

/*return if the vertex was not found*/
if(element==NULL)
return -1;

/*pass back the adjacency list for the vertex*/
*adjlist=list_data(element);
return 0;
}


int graph_is_adjacent(const Graph *graph,const void *data1,const void *data2)
{
ListElmt *element,*prev;

/*locate the adjacency list for the first vertex*/
prev=NULL;
for(element=list_head(&graph->adjlists);element !=NULL;element=list_next(element))
{
if(graph->match(data1,((AdjList *)list_data(element))->vertex))
break;
prev=element;
}

/*return if the vertex was not found*/
if(element==NULL)
return 0;

/*return whether the second vertex is in the adjacency list of the first*/
return set_is_member(&((AdjList *)list_data(element))->adjacent,data2);
}


 

应用实例:

兴许补上