关键路径(CriticalPath)算法
#include <stdio.h>
#include <stdlib.h>
#include <malloc.h> #define MAXVEX 30 //最大顶点数
#define MAXEDGE 30 //最大边数
#define INFINITY 65535 //∞ //定义全局变量
int *etv, *ltv;//事情最早发生和最迟发生指针数组
int *stack2;//用于存储拓扑排序的栈
int top2;//用于指向stack2栈指针 /* 邻接矩阵结构 */
typedef struct
{
int vexs[MAXVEX];//顶点下标
int arc[MAXVEX][MAXVEX];//矩阵(路径)
int numVertexes, numEdges;//当前图中的顶点数和边数
}MGraph; /* 邻接表结构 */
typedef struct EdgeNode //
{//边表结点
int adjvex;//顶点下标
int weight;//路径
struct EdgeNode *next;//指向想一个边表结点
}EdgeNode; typedef struct VertexNode
{//顶点结点
int in;//入度
int data;//顶点信息
EdgeNode *firstedge;//指向边表头指针
}VertexNode, AdjList[MAXVEX]; typedef struct
{
AdjList adjList;//顶点向量
int numVertexes, numEdges;//顶点数和边数
}graphAdjList, *GraphAdjList; void CreateMGraph(MGraph *G)
{/* 构建图 */
int i, j; // printf("请输入顶点数和边数:\n");
G->numVertexes = ;
G->numEdges = ; //初始化顶点下标
for(i=; i<G->numVertexes; i++)
G->vexs[i] = i; //初始化矩阵
for(i=; i<G->numVertexes; i++)
for(j=; j<G->numVertexes; j++)
if(i == j)
G->arc[i][j] = ;
else
G->arc[i][j] = INFINITY; //内置输入
G->arc[][] = ;
G->arc[][] = ;
G->arc[][] = ;
G->arc[][] = ;
G->arc[][] = ;
G->arc[][] = ;
G->arc[][] = ;
G->arc[][] = ;
G->arc[][] = ;
G->arc[][] = ;
G->arc[][] = ;
G->arc[][] = ;
G->arc[][] = ; return ;
} void CreateALGraph(MGraph G, GraphAdjList *GL)
{/* 利用邻接矩阵,构建邻接表 */
int i, j;
EdgeNode *e; *GL = (GraphAdjList)malloc(sizeof(graphAdjList));//*GL代表主函数的GL指向
(*GL)->numVertexes = G.numVertexes;/* 读取信息 */
(*GL)->numEdges = G.numEdges; //初始化
for(i=; i<G.numVertexes; i++)
{
(*GL)->adjList[i].in = ;
(*GL)->adjList[i].data = G.vexs[i];//读取顶点下标
(*GL)->adjList[i].firstedge = NULL;
} //构建邻接表
for(i=; i<G.numVertexes; i++)
for(j=; j<G.numVertexes; j++)
if( != G.arc[i][j] && INFINITY > G.arc[i][j])
{//若存在路径
e = (EdgeNode *)malloc(sizeof(EdgeNode));//申请
e->adjvex = j;//存顶点
e->weight = G.arc[i][j];//存路径
e->next = (*GL)->adjList[i].firstedge;//存表头指针
(*GL)->adjList[i].firstedge = e;//头插
(*GL)->adjList[j].in ++;//顶点入度+1
}
return ;
} void TopologicalSort(GraphAdjList GL)
{/* 拓扑排序 */
EdgeNode *e;
int i, gettop, k;
int top = ;//用于指向stack栈顶
int count = ;//技术输出
int *stack;//建栈存储入度位0的顶点
stack = (int *)malloc(sizeof(int));
for(i=; i<GL->numVertexes; i++)//把所有入度位0的顶点如stack栈中
if( == GL->adjList[i].in)
stack[++top] = i; etv = (int *)malloc(sizeof(int) * GL->numVertexes);//最早发生etv指针申请空间
for(i=; i<GL->numVertexes; i++)//初始化最早发生etv数组
etv[i] = ; stack2 = (int *)malloc(sizeof(int) * GL->numVertexes);//存储拓扑排序序列
top2 = ;//用于指向stack2栈顶
printf("Topological:\t");
while( != top)
{//若有顶点入度为0
gettop = stack[top--];//出栈
printf("%3d->", GL->adjList[gettop].data);//输出出栈的栈顶的顶点信息
count ++;//输出记数 stack2[++top2] = gettop;//出栈元素赋给存储拓扑序列栈 for(e=GL->adjList[gettop].firstedge; e; e=e->next)
{//出栈的栈顶元素若有邻接点
k = e->adjvex;//邻接点-顶点下标
if(! (--GL->adjList[k].in))//邻接点顶点下标-1(gettop已指向),是否入度位0?
stack[++top] = k;//是则入stack的栈 /*出栈的最早发生时间值+出栈邻接表的路径 > 出栈邻接表顶点下标的最早发生时间值
就是v0-v1-v3和v0-v2-v3都是从起点到达汇点,3哪一个路径较远 */
if((etv[gettop] + e->weight) > etv[k])
etv[k] = etv[gettop] + e->weight;
}
}
printf("\n"); if(count < GL->numVertexes)//若有环,则结束程序
exit(-); return ;
} void CriticalPath(GraphAdjList GL)
{/* 关键路径 */
EdgeNode *e;
int i, gettop, k, j;
int ete, lte;//最早发生和最迟发生的变量 TopologicalSort(GL);//调用拓扑排序函数 ltv = (int *)malloc(sizeof(int) * GL->numVertexes);//最迟发生指针边表指向申请空间
for(i=; i<GL->numVertexes; i++)//初始化最迟发生数组
ltv[i] = etv[GL->numVertexes - ];//起到汇最大值 //输出最早发生数组
printf("etv:\t\t");
for(i=; i<GL->numVertexes; i++)
printf("%3d->", etv[i]);
printf("\n"); while( != top2)
{//存放拓扑排序数列
gettop = stack2[top2--];
for(e=GL->adjList[gettop].firstedge; e; e=e->next)
{
k = e->adjvex;
if((ltv[k] - e->weight) < ltv[gettop])
ltv[gettop] = ltv[k] - e->weight;
/*最晚发生时间值(出stack2(存拓扑排序)栈元素的邻接点顶点下标) - 邻接路径 < 最晚发生数组【出栈元素0】
用拓扑排序的最长路径 - 邻接路径 < 最晚发生值[出栈元素]*/
}
} printf("ltv:\t\t");//输出最晚值数组
for(i=; i<GL->numVertexes; i++)
printf("%3d->", ltv[i]);
printf("\n"); for(j=; j<GL->numVertexes; j++)
for(e=GL->adjList[j].firstedge; e; e=e->next)
{
k = e->adjvex;
ete = etv[j];//最早发生时间
lte = ltv[k] - e->weight;//最迟发生时间 if(ete == lte)//相等即在关键路径上
printf("<v%d - v%d> length : %d \n", GL->adjList[j].data, GL->adjList[k].data, e->weight);
} return ;
} int main(void) {
MGraph G;
GraphAdjList GL;
system("title 关键路径");
CreateMGraph(&G);
CreateALGraph(G, &GL);
CriticalPath(GL); return ;
}
关键路径(CriticalPath)算法的更多相关文章
- 关键路径——CriticalPath算法
背景:本文是在小甲鱼数据结构教学视频中的代码的基础上,添加详细注释而完成的.该段代码并不完整,仅摘录了核心算法部分,结合自己的思考,谈谈理解. 关键路径: 即决定一项工程的完成时间的路径. 如下图所示 ...
- 算法与数据结构(八) AOV网的关键路径
上篇博客我们介绍了AOV网的拓扑序列,请参考<数据结构(七) AOV网的拓扑排序(Swift面向对象版)>.拓扑序列中包括项目的每个结点,沿着拓扑序列将项目进行下去是肯定可以将项目完成的, ...
- 算法与数据结构(八) AOV网的关键路径(Swift版)
上篇博客我们介绍了AOV网的拓扑序列,请参考<数据结构(七) AOV网的拓扑排序(Swift面向对象版)>.拓扑序列中包括项目的每个结点,沿着拓扑序列将项目进行下去是肯定可以将项目完成的, ...
- 图->有向无环图->求关键路径
文字描述 与AOV-网相对应的是AOE-网(Activity on Edge)即边表示活动的网.AOE-网是一个带权的有向无环图.其中,顶点表示事件Event,弧表示活动,权表示活动持续的时间.通常, ...
- AOE网与关键路径简介
前面我们说过的拓扑排序主要是为解决一个工程能否顺序进行的问题,但有时我们还需要解决工程完成需要的最短时间问题.如果我们要对一个流程图获得最短时间,就必须要分析它们的拓扑关系,并且找到当中最关键的流程, ...
- java数据结构_笔记(5)_图的算法
图的算法 1 图的遍历图的遍历就是从图中某个顶点出发,按某种方法对图中所有顶点访问且仅访问一次.遍历算法是求解图的连通性问题.拓扑排序和求关键路径等算法的基础. 2 深度优先遍历从图中某个顶点V 出发 ...
- MindFusion 中节点关键路径的遍历
工作中总能遇到 一些 奇葩的需求,提出这些奇葩需求的人,多半也是奇葩的人,要么不懂相关的计算机软件知识,要么就是瞎扯蛋,异想天开,然而这些奇葩的需求,我也总能碰到.言规正传,在一次项目中,使用了 Mi ...
- 数据结构(c语言版)代码
第1章 绪论 文档中源码及测试数据存放目录:数据结构\▲课本算法实现\▲01 绪论 概述 第一章作为绪论,主要介绍了数据结构与算法中的一些基本概念和术语.对于这些概念术语 ...
- 返回数据方法DeaCacheCommand,由CRL自动实现
越来越多的人学起了前端,或许部分的初衷仅是它简单易上手以及好找工作,毕竟几年前只会个html和css就能有工作,悄悄告诉泥萌,这也是博主一年前的初衷 还好numpy, scikit-learn都提供了 ...
随机推荐
- Shell变量替换,命令替换,转义字符
如果表达式中包含特殊字符,Shell 将会进行替换.例如,在双引号中使用变量就是一种替换,转义字符也是一种替换. 举个例子: #!/bin/bash a=10 echo -e "Value ...
- CSS发光边框文本框效果
7,166 次阅读 ‹ NSH Blog 网页设计 CSS发光边框文本框效果 或许你看过Safari浏览器下,任何输入框都会有一个发光的蓝色边框,这不单纯只是蓝色边框而已,其实包含了许多CSS3技巧知 ...
- java_IO流之 NIO
NIO 定义 即新IO,在JDK1.4的java.nio.*包中引入,其目的在于提高速度. 在Java1.4之前的I/O系统中,提供的都是面向流的I/O系统,系统一次一个字节地处理数据,一个输入流产生 ...
- C++ 线程的创建,挂起,唤醒,终止
例子: 线程代码: DWORD __stdcall ThreadProc(LPVOID lpParameter) { CMultiThreadDlg * pdlg = (CMultiThreadDlg ...
- autoprefixer安装或者里sass的$mixin处理浏览器前缀
Autoprefixer是一个后处理程序,不象Sass以及Stylus之类的预处理器.它适用于普通的CSS,可以实现css3代码自动补全.也可以轻松跟Sass,LESS及Stylus集成,在CSS编译 ...
- IOS 排序算法
/** * @brief 冒泡排序法 * * @param arr 需要排序的数组 */ -(void)BubbleSort:(NSMutableArray *)arr { // 取第一个与其邻接的对 ...
- C#常用語法糖(Csharp Syntactic sugar)
首先需要声明的是“语法糖”这个词绝非贬义词,它可以给我带来方便,是一种便捷的写法,编译器会帮我们做转换:而且可以提高开发编码的效率,在性能上也不会带来损失.这让java开发人员羡慕不已,呵呵. 1. ...
- ubuntu中文实训手册
http://people.ubuntu.com/~happyaron/udc-cn/lucid-html/ http://www.apachefriends.org/zh_cn/xampp-linu ...
- 使用Xcode上传代码至GitHub
几乎所有iOS程序员都上过GitHub寻找开源类库,的确,GitHub上有大量优秀的开源类库供大家学习.但是如何在Xcode中上传代码至GitHub呢? (开始之前先安装git,具体方法这里讲的很清楚 ...
- java学习之部分笔记2
1.变量 实例变量和局部变量 实例变量系统会自动初始化为0和null(string),局部变量必须设定初始值. 静态方法里只能引用静态变量 数据类型的自动转换! int—>long 2.构造方法 ...