一、个人理解-简单说说


关键路径其实就是一个点到另外一个点的最唱路径,其中这个图是有向无环图。求出工程里面最关键的事件,不能拖延的事件路径,参考P183-P186。解决的思路,便是找出每个事件点的最早开始时间Ve和最晚完成时间ee,当Ve==ee的情况下就是不能拖延的事件,便是关键路径。



二、数据结构与算法

1、有向图的十字链表 


//==========有向图的十字链表===============
typedef int InfoType;
typedef struct ArcBox{
	int tailvex, headvex;//数组索引的弧头和弧尾
	struct ArcBox *hlink, *tlink;//头链和尾链 头链是弧头相同的下一条弧 尾弧是弧尾相同的一条弧
	InfoType info;
}ArcBox;//弧

typedef struct VexNode{
	VertexType data;
	ArcBox *firstin, *firstout;
}VexNode;//头结点

typedef struct{
	VexNode xlist[MAX_VERTEX_NUM];
	int vexnum, arcnum;//顶点总数 和弧总数
}OLGraph;

Status CreateDG(OLGraph &G, int vexnum, char * vexs, int arcnum, int maps[][6]){
	//创建有向图
	G.vexnum = vexnum;
	G.arcnum = arcnum;
	for (int i = 0; i < G.vexnum; i++)
	{
		//初始化指针
		G.xlist[i].data = vexs[i];
		G.xlist[i].firstin = NULL;
		G.xlist[i].firstout = NULL;
	}
	for (int i = 0; i < vexnum; i++)
	{
		for (int j = 0; j < vexnum; j++)
		{
			if (maps[i][j] != 0)
			{
				//i到j这个有有弧
				ArcBox * p = (ArcBox*)malloc(sizeof(ArcBox));
				
				*p = { i, j, G.xlist[j].firstin, G.xlist[i].firstout, maps[i][j] };
				G.xlist[i].firstout = p;
				G.xlist[j].firstin = p;
			}
		}
	}
	return OK;
}
//对弧的所有遍历.
void PrintfGraphDg(OLGraph G){
	printf("弧尾\t弧头\n");
	for (int i = 0; i < G.vexnum; i++)
	{
		ArcBox* arc = G.xlist[i].firstout;
		while (arc!=NULL)
		{
			printf("%d\t%d \t%c->%c 权重:%d\n", arc->tailvex, arc->headvex, G.xlist[arc->tailvex].data, G.xlist[arc->headvex].data,arc->info);
			arc = arc->tlink;
		}
	}
}

2、关键路径算法



//利用拓扑排序算法计算ve数组
int ve[MAX_VERTEX_NUM];//点的最早完成时间数组
int vl[MAX_VERTEX_NUM];//点的最迟完成时间数组
Status TopologicalOrder(OLGraph G,SqStack &T){
	//初始化ve数组
	for (int i = 0; i < G.vexnum; i++)
	{
		ve[i] = 0; 
	}
	int indegree[MAX_VERTEX_NUM];//存放各顶点入度的数组
	//生成入度数组
	for (int i = 0; i < G.vexnum; i++)
	{
		int count = 0;
		ArcBox* arc = G.xlist[i].firstin;//入度的弧
		while (arc != NULL)
		{
			count++;
			arc = arc->hlink;
		}
		indegree[i] = count;
		printf("%c的入度为%d \n", G.xlist[i].data, indegree[i]);
	}

	SqStack stack;//保存的是入度为0的结点
	InitStack(stack);
	InitStack(T);//初始化栈
	for (int i = 0; i < G.vexnum; i++)
	{
		if (indegree[i] == 0)
		{
			//入度为0 进栈
			//printf("push %d\n", i);
			Push(stack, i);
		}
	}

	int count = 0;//计数计算拓扑排序入度为0 的个数
	while (!StackEmpty(stack)){
		//在栈不为空的情况下
		int elem;
		Pop(stack, elem);
		//获取逆的拓扑排序
		Push(T, elem);

		count++;
		printf("pop %d\n", elem);
		//printf("%c", G.xlist[elem].data);

		//遍历所有指向elem顶点的弧
		ArcBox* arc = G.xlist[elem].firstout;//出去的弧
		while (arc != NULL)
		{
			int k = arc->headvex;//当前弧的头的索引
			if ((--indegree[k]) == 0)
			{
				//printf("push %d\n", arc->tailvex);
				Push(stack, k);
			}
			//找出j-k的弧上最大的边
			if (ve[elem] + arc->info > ve[k])
			{
				ve[k] = ve[elem] + arc->info;
			}
			arc = arc->tlink;
		}
	}
	if (count < G.vexnum)
	{
		printf("个图有回路");
		return ERROR;//这个图有回路
	}


	return OK;
}
//求出关键路径
Status CriticalPath(OLGraph G){
	SqStack T;
	TopologicalOrder(G, T);
	for (int i = 0; i < G.vexnum; i++)
	{
		//printf("%d:",ve[i]);
	}
	//计算出vl数组
	//初始化vl数组 最后一个点的ve与vl相等
	for (int i = 0; i < G.vexnum; i++)
	{
		vl[i] = ve[G.vexnum - 1];
	}
	//使用逆拓扑排序计算出所有的vl
	while (!StackEmpty(T))
	{
		int j;//当前计算的点j
		Pop(T, j);
		//遍历点j的所有
		ArcBox* arc = G.xlist[j].firstout;//出去的弧
		while (arc != NULL)
		{
			int k = arc->headvex;//当前弧的头的索引
			//找出j-k的弧上最大的边
			if (vl[k] - arc->info < vl[j])
			{
				vl[j] = vl[k] - arc->info;
			}
			arc = arc->tlink;
		}
	}

	//for (int i = 0; i < G.vexnum; i++)
	//{
	//	printf("%d:", vl[i]);
	//}

	//求出ee和el 就是关于弧的最早开始时间和最晚开始时间 遍历所有的弧
	for (int i = 0; i < G.vexnum; i++)
	{
		ArcBox* arc = G.xlist[i].firstout;
		while (arc != NULL)
		{
			//printf("%d\t%d \t%c->%c 权重:%d\n", arc->tailvex, arc->headvex, G.xlist[arc->tailvex].data, G.xlist[arc->headvex].data, arc->info);
			int ee = ve[arc->tailvex];
			int el = vl[arc->headvex] - arc->info;
			if (ee == el)
			{
				printf("%d\t%d \t%c->%c 权重:%d\n", arc->tailvex, arc->headvex, G.xlist[arc->tailvex].data, G.xlist[arc->headvex].data, arc->info);
			}
			arc = arc->tlink;
		}
	}
	return OK;
}



三、执行



//链表创建进行拓扑排序和关键路径
	int vexnum = 6;
	char *vexs = "abcdef";
	int arcnum = 8;
	int maps[6][6] = {
		{ 0, 3, 2, 0, 0, 0 },
		{ 0, 0, 0, 2, 3, 0 },
		{ 0, 0, 0, 4, 0, 3 },
		{ 0, 0, 0, 0, 0, 2 },
		{ 0, 0, 0, 0, 0, 1 },
		{ 0, 0, 0, 0, 0, 0 }
	};
	OLGraph G;
	CreateDG(G,vexnum,vexs,arcnum,maps);
	//PrintfGraphDg(G);
	//DFSTraverse_L(G);
	//BFSTraverse_L(G);
	//TopologicalSort(G);//拓扑排序
	CriticalPath(G);//关键路径



输出:



a的入度为0
b的入度为1
c的入度为1
d的入度为2
e的入度为1
f的入度为3
pop 0
pop 1
pop 4
pop 2
pop 3
pop 5
0       2       a->c 权重:2
2       3       c->d 权重:4
3       5       d->f 权重:2
请按任意键继续. . .