第十二周项目1 - 验证算法(1)(2)(3)(4)(5)

/*  

*Copyright  (c)2017,烟台大学计算机与控制工程学院      

*All rights reservrd.      

*文件名称 :test.cpp      

*作者:潘亚楠  

*完成时间:2017年12月15日      

*版本号:v1.0      

*问题描述: 验证算法

问题及代码:

程序中graph.h是图存储结构的“算法库”中的头文件,详情请单击链接…

(1)Prim算法的验证

#include <stdio.h>  
#include <malloc.h>  
#include "graph.h"  
  
void Prim(MGraph g,int v)  
{  
    int lowcost[MAXV];          //顶点i是否在U中  
    int min;  
    int closest[MAXV],i,j,k;  
    for (i=0; i<g.n; i++)           //给lowcost[]和closest[]置初值  
    {  
        lowcost[i]=g.edges[v][i];  
        closest[i]=v;  
    }  
    for (i=1; i<g.n; i++)           //找出n-1个顶点  
    {  
        min=INF;  
        for (j=0; j<g.n; j++)     //在(V-U)中找出离U最近的顶点k  
            if (lowcost[j]!=0 && lowcost[j]<min)  
            {  
                min=lowcost[j];  
                k=j;            //k记录最近顶点的编号  
            }  
        printf(" 边(%d,%d)权为:%d\n",closest[k],k,min);  
        lowcost[k]=0;           //标记k已经加入U  
        for (j=0; j<g.n; j++)       //修改数组lowcost和closest  
            if (g.edges[k][j]!=0 && g.edges[k][j]<lowcost[j])  
            {  
                lowcost[j]=g.edges[k][j];  
                closest[j]=k;  
            }  
    }  
}  
  
int main()  
{  
    MGraph g;  
    int A[6][6]=  
    {  
        {0,6,1,5,INF,INF},  
        {6,0,5,INF,3,INF},  
        {1,5,0,5,6,4},  
        {5,INF,5,0,INF,2},  
        {INF,3,6,INF,0,6},  
        {INF,INF,4,2,6,0}  
    };  
    ArrayToMat(A[0], 6, g);  
    printf("最小生成树构成:\n");  
    Prim(g,0);  
    return 0;  
}  
(2)Kruskal算法的验证


#include <stdio.h>  
#include <malloc.h>  
#include "graph.h"  
#define MaxSize 100  
typedef struct  
{  
    int u;     //边的起始顶点  
    int v;     //边的终止顶点  
    int w;     //边的权值  
} Edge;  
  
void InsertSort(Edge E[],int n) //对E[0..n-1]按递增有序进行直接插入排序  
{  
    int i,j;  
    Edge temp;  
    for (i=1; i<n; i++)  
    {  
        temp=E[i];  
        j=i-1;              //从右向左在有序区E[0..i-1]中找E[i]的插入位置  
        while (j>=0 && temp.w<E[j].w)  
        {  
            E[j+1]=E[j];    //将关键字大于E[i].w的记录后移  
            j--;  
        }  
        E[j+1]=temp;        //在j+1处插入E[i]  
    }  
}  
  
void Kruskal(MGraph g)  
{  
    int i,j,u1,v1,sn1,sn2,k;  
    int vset[MAXV];  
    Edge E[MaxSize];    //存放所有边  
    k=0;                //E数组的下标从0开始计  
    for (i=0; i<g.n; i++)   //由g产生的边集E  
        for (j=0; j<g.n; j++)  
            if (g.edges[i][j]!=0 && g.edges[i][j]!=INF)  
            {  
                E[k].u=i;  
                E[k].v=j;  
                E[k].w=g.edges[i][j];  
                k++;  
            }  
    InsertSort(E,g.e);      //采用直接插入排序对E数组按权值递增排序  
    for (i=0; i<g.n; i++)   //初始化辅助数组  
        vset[i]=i;  
    k=1;    //k表示当前构造生成树的第几条边,初值为1  
    j=0;    //E中边的下标,初值为0  
    while (k<g.n)       //生成的边数小于n时循环  
    {  
        u1=E[j].u;  
        v1=E[j].v;      //取一条边的头尾顶点  
        sn1=vset[u1];  
        sn2=vset[v1];   //分别得到两个顶点所属的集合编号  
        if (sn1!=sn2)   //两顶点属于不同的集合  
        {  
            printf("  (%d,%d):%d\n",u1,v1,E[j].w);  
            k++;                     //生成边数增1  
            for (i=0; i<g.n; i++)   //两个集合统一编号  
                if (vset[i]==sn2)   //集合编号为sn2的改为sn1  
                    vset[i]=sn1;  
        }  
        j++;               //扫描下一条边  
    }  
}  
  
int main()  
{  
    MGraph g;  
    int A[6][6]=  
    {  
        {0,6,1,5,INF,INF},  
        {6,0,5,INF,3,INF},  
        {1,5,0,5,6,4},  
        {5,INF,5,0,INF,2},  
        {INF,3,6,INF,0,6},  
        {INF,INF,4,2,6,0}  
    };  
    ArrayToMat(A[0], 6, g);  
    printf("最小生成树构成:\n");  
    Kruskal(g);  
    return 0;  
}  
(3)Dijkstra算法的验证


#include <stdio.h>  
#include <malloc.h>  
#include "graph.h"  
#define MaxSize 100  
void Ppath(int path[],int i,int v)  //前向递归查找路径上的顶点  
{  
    int k;  
    k=path[i];  
    if (k==v)  return;          //找到了起点则返回  
    Ppath(path,k,v);            //找顶点k的前一个顶点  
    printf("%d,",k);            //输出顶点k  
}  
void Dispath(int dist[],int path[],int s[],int n,int v)  
{  
    int i;  
    for (i=0; i<n; i++)  
        if (s[i]==1)  
        {  
            printf("  从%d到%d的最短路径长度为:%d\t路径为:",v,i,dist[i]);  
            printf("%d,",v);    //输出路径上的起点  
            Ppath(path,i,v);    //输出路径上的中间点  
            printf("%d\n",i);   //输出路径上的终点  
        }  
        else  printf("从%d到%d不存在路径\n",v,i);  
}  
void Dijkstra(MGraph g,int v)  
{  
    int dist[MAXV],path[MAXV];  
    int s[MAXV];  
    int mindis,i,j,u;  
    for (i=0; i<g.n; i++)  
    {  
        dist[i]=g.edges[v][i];      //距离初始化  
        s[i]=0;                     //s[]置空  
        if (g.edges[v][i]<INF)      //路径初始化  
            path[i]=v;  
        else  
            path[i]=-1;  
    }  
    s[v]=1;  
    path[v]=0;              //源点编号v放入s中  
    for (i=0; i<g.n; i++)               //循环直到所有顶点的最短路径都求出  
    {  
        mindis=INF;                 //mindis置最小长度初值  
        for (j=0; j<g.n; j++)       //选取不在s中且具有最小距离的顶点u  
            if (s[j]==0 && dist[j]<mindis)  
            {  
                u=j;  
                mindis=dist[j];  
            }  
        s[u]=1;                     //顶点u加入s中  
        for (j=0; j<g.n; j++)       //修改不在s中的顶点的距离  
            if (s[j]==0)  
                if (g.edges[u][j]<INF && dist[u]+g.edges[u][j]<dist[j])  
                {  
                    dist[j]=dist[u]+g.edges[u][j];  
                    path[j]=u;  
                }  
    }  
    Dispath(dist,path,s,g.n,v);     //输出最短路径  
}  
  
int main()  
{  
    MGraph g;  
    int A[7][7]=  
    {  
        {0,4,6,6,INF,INF,INF},  
        {INF,0,1,INF,7,INF,INF},  
        {INF,INF,0,INF,6,4,INF},  
        {INF,INF,2,0,INF,5,INF},  
        {INF,INF,INF,INF,0,INF,6},  
        {INF,INF,INF,INF,1,0,8},  
        {INF,INF,INF,INF,INF,INF,0}  
    };  
    ArrayToMat(A[0], 7, g);  
    Dijkstra(g,0);  
    return 0;  
}  

(4)Floyd算法验证

#include <stdio.h>  
#include <malloc.h>  
#include "graph.h"  
#define MaxSize 100  
void Ppath(int path[][MAXV],int i,int j)  //前向递归查找路径上的顶点  
{  
    int k;  
    k=path[i][j];  
    if (k==-1) return;  //找到了起点则返回  
    Ppath(path,i,k);    //找顶点i的前一个顶点k  
    printf("%d,",k);  
    Ppath(path,k,j);    //找顶点k的前一个顶点j  
}  
void Dispath(int A[][MAXV],int path[][MAXV],int n)  
{  
    int i,j;  
    for (i=0; i<n; i++)  
        for (j=0; j<n; j++)  
        {  
            if (A[i][j]==INF)  
            {  
                if (i!=j)  
                    printf("从%d到%d没有路径\n",i,j);  
            }  
            else  
            {  
                printf("  从%d到%d=>路径长度:%d 路径:",i,j,A[i][j]);  
                printf("%d,",i);    //输出路径上的起点  
                Ppath(path,i,j);    //输出路径上的中间点  
                printf("%d\n",j);   //输出路径上的终点  
            }  
        }  
}  
void Floyd(MGraph g)  
{  
    int A[MAXV][MAXV],path[MAXV][MAXV];  
    int i,j,k;  
    for (i=0; i<g.n; i++)  
        for (j=0; j<g.n; j++)  
        {  
            A[i][j]=g.edges[i][j];  
            path[i][j]=-1;  
        }  
    for (k=0; k<g.n; k++)  
    {  
        for (i=0; i<g.n; i++)  
            for (j=0; j<g.n; j++)  
                if (A[i][j]>A[i][k]+A[k][j])  
                {  
                    A[i][j]=A[i][k]+A[k][j];  
                    path[i][j]=k;  
                }  
    }  
    Dispath(A,path,g.n);   //输出最短路径  
}  
int main()  
{  
    MGraph g;  
    int A[4][4]=  
    {  
        {0,  5,INF,7},  
        {INF,0,  4,2},  
        {3,  3,  0,2},  
        {INF,INF,1,0}  
    };  
    ArrayToMat(A[0], 4, g);  
    Floyd(g);  
    return 0;  
}  
(5)拓扑排序算法验证

#include <stdio.h>  
#include <malloc.h>  
#include "graph.h"  
  
void TopSort(ALGraph *G)  
{  
    int i,j;  
    int St[MAXV],top=-1;            //栈St的指针为top  
    ArcNode *p;  
    for (i=0; i<G->n; i++)          //入度置初值0  
        G->adjlist[i].count=0;  
    for (i=0; i<G->n; i++)          //求所有顶点的入度  
    {  
        p=G->adjlist[i].firstarc;  
        while (p!=NULL)  
        {  
            G->adjlist[p->adjvex].count++;  
            p=p->nextarc;  
        }  
    }  
    for (i=0; i<G->n; i++)  
        if (G->adjlist[i].count==0)  //入度为0的顶点进栈  
        {  
            top++;  
            St[top]=i;  
        }  
    while (top>-1)                  //栈不为空时循环  
    {  
        i=St[top];  
        top--;              //出栈  
        printf("%d ",i);            //输出顶点  
        p=G->adjlist[i].firstarc;   //找第一个相邻顶点  
        while (p!=NULL)  
        {  
            j=p->adjvex;  
            G->adjlist[j].count--;  
            if (G->adjlist[j].count==0)//入度为0的相邻顶点进栈  
            {  
                top++;  
                St[top]=j;  
            }  
            p=p->nextarc;       //找下一个相邻顶点  
        }  
    }  
}  
  
int main()  
{  
    ALGraph *G;  
    int A[7][7]=  
    {  
        {0,0,1,0,0,0,0},  
        {0,0,0,1,1,0,1},  
        {0,0,0,1,0,0,0},  
        {0,0,0,0,1,1,0},  
        {0,0,0,0,0,0,0},  
        {0,0,0,0,0,0,0},  
        {0,0,0,0,0,1,0}  
    };  
    ArrayToList(A[0], 7, G);  
    DispAdj(G);  
    printf("\n");  
    printf("拓扑序列:");  
    TopSort(G);  
    printf("\n");  
    return 0;  
}  

运行结果:






  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
图(Graph)是由顶点的有穷非空集合和顶点之间边的集合组成,通常表示为:G(V,E),其中,G表示一个图,V是图G中顶点的集合,E是图G中边的集合。在图中的数据元素,我们称之为顶点(Vertex),顶点集合有穷非空。在图中,任意两个顶点之间都可能有关系,顶点之间的逻辑关系用边来表示,边集可以是空的。 图按照边的有无方向分为无向图和有向图。无向图由顶点和边组成,有向图由顶点和弧构成。弧有弧尾和弧头之分,带箭头一端为弧头。 图按照边或弧的多少分稀疏图和稠密图。如果图中的任意两个顶点之间都存在边叫做完全图,有向的叫有向完全图。若无重复的边或顶点到自身的边则叫简单图。 图中顶点之间有邻接点、依附的概念。无向图顶点的边数叫做度。有向图顶点分为入度和出度。 图上的边或弧带有权则称为网。 图中顶点间存在路径,两顶点存在路径则说明是连通的,如果路径最终回到起始点则称为环,当中不重复的叫简单路径。若任意两顶点都是连通的,则图就是连通图,有向则称为强连通图。图中有子图,若子图极大连通则就是连通分量,有向的则称为强连通分量。 无向图中连通且n个顶点n-1条边称为生成树。有向图中一顶点入度为0其余顶点入度为1的叫有向树。一个有向图由若干棵有向树构成生成森林。
《数据结构》实验教学大纲 数据结构实验 Data Structure Experiment 工学 计算机科学与技术 先修课程:高等数学、离散数学、程序设计基础(C语言 或 C++语言) 课程性质 数据结构是计算机科学的算法理论基础和软件设计的技术基础,是计算机科学技术专业的基础理论课程,是计算机学科的核心课程之一。在计算机科学技术的各个领域,选择合适的数据结构是一个重要问题;具备分析算法复杂度、比较算法性能和优化算法的能力是计算机专业学生必须具备的重要专业能力。通过数据结构与算法的学习,能进一步提高软件设计与编写高效程序的能力,提高应用计算机技术解决实际问题的能力。 本课程是结合《数据结构》课堂教学安排的实验与实践课程,它是对学生的一种全面综合训练,是与课堂教学与课后练习,完成程序分析与设计、理论与实践相结合的训练的必不可少的一个教学环节。本实验课程目的是加深对数据结构与算法的理解,加强理论与实践的结合,培养学生的综合动手能力。本实验强调基础知识与实际应用相结合,促使学生掌握知识并应用于解决实际问题,培养学生的动手能力和实践应用能力,起到深化理解和灵活掌握教学内容的目的。 课程任务 进行本课程实验之前,课堂任课教师或实验教师必须要求学生认真复习C语言(或C++语言)的基本编程方法,熟悉编程环境。通过本课程实验,使学生学会和掌握本课程的基本知识点和重点内容,理解数据结构的基本概念和基本原理,深刻理解逻辑结构、存储结构、算法设计之间的关系,掌握分析问题的基本方法,熟练编程的基本方法和技巧,提高解决问题的能力。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值