图的邻接表操作

图的邻接表操作,可以进行基本的创建、深度遍历等……也有广度遍历程序,有待提高,有兴趣的你可以试一下…………

/*win-tc
如果你是用VC++的请你去掉getch()和clrscr()涵数*/
#include"stdio.h"
#include"stdlib.h"
#include"conio.h"
#include"malloc.h"
#define NULL 0
#define VertexNum 5     /*用户自定义顶点的大小*/

int visited[VertexNum];

typedef int VertexType;
typedef enum{ERROR=0,OK=1}Status;
typedef struct aduNode
{
    int vertexPos;
    float weight;
    struct adjNode *next;
}AdjNode,*pAdjList;

/**/
typedef struct VertexNode
{
    VertexType vertex;
    pAdjList firstAdj;
}VertexNode,Graph;

/*插入边*/
void InsertEdge(Graph *graph,int flag,VertexType source,VertexType destination,float weight)
{
    int sourcePos,desPos;
    AdjNode* newnode;
    AdjNode *pvertex=NULL,*p=NULL;
    /*求与边关联的两个顶点在顶点表中的位置*/
    sourcePos=GetPos(graph,source);
    desPos=GetPos(graph,destination);
    if(sourcePos==-1||desPos==-1)
    {
        printf("/n***Erroe***:Out of range!/n");
        return;
    }
    else
    {
        newnode=(AdjNode*)malloc(sizeof(AdjNode));
        newnode->vertexPos=desPos;
        newnode->weight=weight;
        newnode->next=NULL;
        pvertex=graph[sourcePos].firstAdj;
        /*生成顶点的第一个邻接点*/
        if(pvertex==NULL)
            graph[sourcePos].firstAdj=newnode;
        else
        {
            /*生成顶点的其他邻接点*/
            while(pvertex!=NULL)
            {
                p=pvertex;
                pvertex=pvertex->next;
            }
            p->next=newnode;
        }
        /*无向图插入边(a,b)的同时也插入边(b,c)*/
        if(flag==1)
        {
            newnode=(AdjNode*)malloc(sizeof(AdjNode));
            newnode->vertexPos=sourcePos;
            newnode->next=NULL;
            pvertex=graph[desPos].firstAdj;
            /*生成顶点的第一个邻接点*/
            if(pvertex==NULL)
                graph[desPos].firstAdj=newnode;
            else
            {
                /*生成顶点的其他邻接点*/
                while(pvertex!=NULL)
                {
                    p=pvertex;
                    pvertex=pvertex->next;
                }
                p->next=newnode;
            }
        }
    }
}


/*图的生成*/
void create_graph(Graph* graph,int flag)
{
    VertexType vertex;
    VertexType vertex1,vertex2;
    int num=VertexNum;
    int i;
    float weight=0;

    clrscr();         /*清屏涵数*/
    printf("/nTotal number of vertex is %d (0....%d).",num,num-1);
    printf("/nplease input the vertex:");
    /*输入顶点,生成顶点表*/
    for(i=0;i<VertexNum;i++)
    {
        scanf("%d",&vertex);
        graph[i].vertex=vertex;
        graph[i].firstAdj=NULL;
    }

    printf("/nplease input the edge(s),-1,-1 to finish!/n");
    /*输入边,生成邻接表*/
    while(1)
    {
        scanf("%d,%d",&vertex1,&vertex2);
        if(vertex1==-1||vertex2==-1)
            break;
        InsertEdge(graph,flag,vertex1,vertex2,weight);
    }
}

/*顶点定位*/
int GetPos(Graph* graph,VertexType vertex)
{
    int i;
    for(i=0;i<VertexNum;i++)
    {
        if(graph[i].vertex==vertex)
            return i;
    }
    return -1;
}

/*取顶点*/
Status GetVertex(Graph* graph,int pos,VertexType* vertex)
{
    if(pos>=VertexNum)
        return ERROR;
    *vertex=graph[pos].vertex;
    return OK;
}

/*求第一个邻接点*/
Status FirstAdj(Graph* graph,VertexType vertex,VertexType* firstAdj)
{
    int pos;
    AdjNode *pvertex=NULL;
    pos=GetPos(graph,vertex);
    if(pos==-1)
        return ERROR;
    pvertex=graph[pos].firstAdj;
    if(pvertex!=NULL)
    {
        return (GetVertex(graph,pvertex->vertexPos,firstAdj));
    }
    return ERROR;
}

/*求下一个邻接点*/
Status NextAdj(Graph* graph,VertexType vertex,VertexType adj,VertexType* nextAdj)
{
    int vertexPos;
    AdjNode *pvertex=NULL;
    VertexType adjNode;
    Status GetVertex(Graph* graph,int pos,VertexType* vertex);

    vertexPos=GetPos(graph,vertex);

    pvertex=graph[vertexPos].firstAdj;
    while(pvertex!=NULL)
    {
        GetVertex(graph,pvertex->vertexPos,&adjNode);
        if(adjNode==adj&& pvertex->next!=NULL)
        {
            vertexPos=pvertex->next->vertexPos;
            return (GetVertex(graph,vertexPos,nextAdj));
        }
        pvertex=pvertex->next;
    }
    return ERROR;
}

/*图的输出*/
void print_graph(Graph* graph)
{
    int i;
    AdjNode *pvertex=NULL;
    VertexType adj;
    printf("/nGraph's adjcency list is:");
    for(i=0;i<VertexNum;i++)
    {
        printf("/nVertex[%d]:%d-->",i,graph[i].vertex);
        pvertex=graph[i].firstAdj;
        while(pvertex!=NULL)
        {
            GetVertex(graph,pvertex->vertexPos,&adj);
            printf("%d  ",adj);
            pvertex=pvertex->next;

        }
    }
}


/*释放图*/
void free_graph(Graph* graph)
{
    AdjNode *pvertex=NULL,*p=NULL;
    int i;
    for(i=0;i<VertexNum;i++)
    {
        pvertex=graph[i].firstAdj;
        while(pvertex!=NULL)
        {
            p=pvertex;
            pvertex=pvertex->next;
            free(p);
            p=NULL;
        }
    }
}


/*深度优先遍历*/
void dfs(Graph* graph,VertexType vertex)
{
    AdjNode* pointer;
    int pos;
    VertexType v;
    VertexType adjVertex,nextAdj;
    Status s;

    pos=GetPos(graph,vertex);
    visited[pos]=1;
    /*输出顶点*/
    printf("%d  ",vertex);
    s=FirstAdj(graph,vertex,&adjVertex);
    while(s!=ERROR)
    {
        pos=GetPos(graph,adjVertex);
        if(visited[pos]==0)
            /*递归调用,深度优先搜索邻接点*/
            dfs(graph,adjVertex);
        s=NextAdj(graph,vertex,adjVertex,&nextAdj);
        adjVertex=nextAdj;
    }
}

/*         要完成它还需要添加队列的相关涵数,如入队和出队,以及要给队列做定义方可完成功能。
           有兴趣的你可以做一下        
/ 广度优先遍历/
void bfs(Graph *graph,int vertex)
{
    AdjNode* pointer;
    int pos;
    VertexType v;
    VertexType adjVertex,nextAdj;
    Status s;
    / 搜索起始点入队 /
    EnQueue(vertex);
    pos=GetPos(graph,vertex);
    / 置入队标志/
    visited[pos]=1;
    / 判断队列是否为空/
    while(front!=rear)
    {
        vertex=DeQueue();
        / 输出顶点/
        printf("%d",vertex);
        s=FirstAdj(graph,vertex,&adjVertex);
        while(s!=ERROR)
        {
            pos=GetPos(graph,adjVertex);
            / 如果该邻接点没有入队,将其入队/
            if(visited[pos]==0)
            {
                EnQueue(adjVertex);
                visited[pos]=1;
            }
            s=NextAdj(graph,vertex,adjVertex,&nextAdj);
            adjVertex=nextAdj;
        }
    }
}
*/

/**/
void main()
{
    int i;
    Graph graph[VertexNum];

    /*生成图*/
    create_graph(graph,1);
    print_graph(graph);
    /*记录顶点是否被访问过*/
    for(i=0;i<VertexNum;i++)
        visited[i]=0;
    /*深度优先搜索*/
    printf("/nDepth-first-search:/n");
    for(i=0;i<VertexNum;i++)
    {
        if(visited[i]==0)
            dfs(graph,graph[i].vertex);
    }
    free_graph(graph);
    getch();             /*让屏幕停顿涵数*/
}

结果如下:

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值