PTA数据结构与算法题目集(中文) 7-11
PTA数据结构与算法题目集(中文) 7-11
假定一个工程项目由一组子任务构成,子任务之间有的可以并行执行,有的必须在完成了其它一些子任务后才能执行。“任务调度”包括一组子任务、以及每个子任务可以执行所依赖的子任务集。
比如完成一个专业的所有课程学习和毕业设计可以看成一个本科生要完成的一项工程,各门课程可以看成是子任务。有些课程可以同时开设,比如英语和C程序设计,它们没有必须先修哪门的约束;有些课程则不可以同时开设,因为它们有先后的依赖关系,比如C程序设计和数据结构两门课,必须先学习前者。
但是需要注意的是,对一组子任务,并不是任意的任务调度都是一个可行的方案。比如方案中存在“子任务A依赖于子任务B,子任务B依赖于子任务C,子任务C又依赖于子任务A”,那么这三个任务哪个都不能先执行,这就是一个不可行的方案。
任务调度问题中,如果还给出了完成每个子任务需要的时间,则我们可以算出完成整个工程需要的最短时间。在这些子任务中,有些任务即使推迟几天完成,也不会影响全局的工期;但是有些任务必须准时完成,否则整个项目的工期就要因此延误,这种任务就叫“关键活动”。
请编写程序判定一个给定的工程项目的任务调度是否可行;如果该调度方案可行,则计算完成整个工程项目需要的最短时间,并输出所有的关键活动。
输入格式:
输入第1行给出两个正整数N(≤)和M,其中N是任务交接点(即衔接相互依赖的两个子任务的节点,例如:若任务2要在任务1完成后才开始,则两任务之间必有一个交接点)的数量。交接点按1~N编号,M是子任务的数量,依次编号为1~M。随后M行,每行给出了3个正整数,分别是该任务开始和完成涉及的交接点编号以及该任务所需的时间,整数间用空格分隔。
输出格式:
如果任务调度不可行,则输出0;否则第1行输出完成整个工程项目需要的时间,第2行开始输出所有关键活动,每个关键活动占一行,按格式“V->W”输出,其中V和W为该任务开始和完成涉及的交接点编号。关键活动输出的顺序规则是:任务开始的交接点编号小者优先,起点编号相同时,与输入时任务的顺序相反。
输入样例:
7 8
1 2 4
1 3 3
2 4 5
3 4 3
4 5 1
4 6 6
5 7 5
6 7 2
输出样例:
17
1->2
2->4
4->6
6->7
题目分析:拓扑排序的基本应用 这是拓扑排序中的关键路径问题 对于拓扑排序是要利用的每个节点的入度来进行入队 关键路径是在这基础上 在每次入队之前先计算其邻接点的Earliest(用其它名称也可以) 而从末尾开始计算Latest时 要从达到最大时间的那个节点开始 该节点的Latest等于Earliest
对于从末尾开始时 要利用出度来进行入队
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<stdlib.h>
#include<malloc.h>
#define MAXVERTEXNUM 110
#define INIFITY 65535 typedef struct ENode* Edge;
struct ENode
{
int V1, V2;
int Weight;
}; typedef struct AdvjNode* PtrToAdvjNode;
struct AdvjNode
{
int Advj;
int Weight;
PtrToAdvjNode Next;
}; typedef struct VNode
{
PtrToAdvjNode FirstNode;
PtrToAdvjNode LastNode;
int InDegree;
int OutDegree;
}AdjList[MAXVERTEXNUM]; typedef struct GNode* Graph;
struct GNode
{
int Nv;
int Ne;
AdjList G;
}; Graph BuildGraph(int VertexNum)
{
Graph Gra = (Graph)malloc(sizeof(struct GNode));
Gra->Nv = VertexNum;
Gra->Ne = ;
for (int i =; i <=Gra->Nv; i++)
{
Gra->G[i].FirstNode = NULL;
Gra->G[i].LastNode = NULL;
Gra->G[i].OutDegree = ;
Gra->G[i].InDegree = ;
}
return Gra;
} void Insert(Edge E, Graph Gra)
{
PtrToAdvjNode NewNode = (PtrToAdvjNode)malloc(sizeof(struct AdvjNode));
NewNode->Advj = E->V2;
NewNode->Weight = E->Weight;
NewNode->Next = Gra->G[E->V1].FirstNode;
Gra->G[E->V1].FirstNode = NewNode;
Gra->G[E->V2].InDegree++; NewNode = (PtrToAdvjNode)malloc(sizeof(struct AdvjNode));
NewNode->Advj = E->V1;
NewNode->Weight = E->Weight;
NewNode->Next = Gra->G[E->V2].LastNode;
Gra->G[E->V2].LastNode = NewNode;
Gra->G[E->V1].OutDegree++;
} Graph CreateGraph()
{
Edge E = (Edge)malloc(sizeof(struct ENode));
int N, M;
scanf("%d%d", &N, &M);
Graph Gra = BuildGraph(N);
Gra->Ne = M;
for (int i = ; i < Gra->Ne; i++)
{
scanf("%d%d%d", &(E->V1), &(E->V2), &(E->Weight));
Insert(E, Gra);
}
return Gra;
} int Queue[MAXVERTEXNUM];
int Rear = ;
int Front = ;
int Size = ;
int IsEmpty()
{
return Size == ;
}
int Succ(int num)
{
if (num < MAXVERTEXNUM)
return num;
else
return ;
} void EnQueue(int V)
{
Rear = Succ(Rear + );
Queue[Rear] = V;
Size++;
} int DeQueue()
{
int num = Queue[Front];
Front = Succ(Front + );
Size--;
return num;
} void Initilize()
{
Rear = ;
Front = ;
Size = ;
} int Earliest[MAXVERTEXNUM];
int Latest[MAXVERTEXNUM];
int TopOrder[MAXVERTEXNUM];
int InDegree[MAXVERTEXNUM];
int OutDegree[MAXVERTEXNUM];
int S;
int TopSort(Graph Gra)
{
int cnt = ;
for (int i = ; i <= Gra->Nv; i++)
{
Earliest[i] = ;
Latest[i] = INIFITY;
}
for (int i = ; i <=Gra->Nv; i++)
for (PtrToAdvjNode W = Gra->G[i].FirstNode; W; W = W->Next)
InDegree[W->Advj]++;
for (int i =; i <=Gra->Nv; i++)
{
if (!InDegree[i])
{
EnQueue(i);
Earliest[i] = ;
}
}
while (!IsEmpty())
{
int V = DeQueue();
TopOrder[S++] = V;
cnt++;
for (PtrToAdvjNode W = Gra->G[V].FirstNode; W; W = W->Next)
{
if (Earliest[V] + W->Weight > Earliest[W->Advj])
Earliest[W->Advj] = Earliest[V] + W->Weight;
if (--InDegree[W->Advj] == )
EnQueue(W->Advj);
}
} //从最后结束的任务开始
int Max = -;
int V;
for (int i = ; i <= Gra->Nv; i++)
{
if (Earliest[i] > Max)
{
Max = Earliest[i];
V = i;
}
} Initilize();
Latest[V] = Max;
for (int i = ; i <= Gra->Nv; i++)
for (PtrToAdvjNode W = Gra->G[i].LastNode; W; W = W->Next)
OutDegree[W->Advj]++;
for (int i = ; i <= Gra->Nv; i++)
if (!OutDegree[i])
EnQueue(i);
while (!IsEmpty())
{
int V = DeQueue();
for (PtrToAdvjNode W = Gra->G[V].LastNode; W; W = W->Next)
{
if (Latest[V] - W->Weight < Latest[W->Advj])
Latest[W->Advj] = Latest[V] - W->Weight;
if (--OutDegree[W->Advj] == )
EnQueue(W->Advj);
}
} Initilize();
for(int i=;i<Gra->Nv;i++)
for (PtrToAdvjNode W = Gra->G[i].FirstNode; W; W = W->Next)
{
if (Latest[W->Advj] - Earliest[i] - W->Weight == )
{
EnQueue(i);
EnQueue(W->Advj);
}
}
if (cnt == Gra->Nv)
return Max;
else
return ;
} int main()
{
Graph G = CreateGraph();
int sum;
if ((sum=TopSort(G)))
{
printf("%d\n",sum);
while (Size)
{
printf("%d->", DeQueue());
printf("%d\n", DeQueue());
}
}
else
printf("");
return ;
}
PTA数据结构与算法题目集(中文) 7-11的更多相关文章
- PTA数据结构与算法题目集(中文) 7-43字符串关键字的散列映射 (25 分)
PTA数据结构与算法题目集(中文) 7-43字符串关键字的散列映射 (25 分) 7-43 字符串关键字的散列映射 (25 分) 给定一系列由大写英文字母组成的字符串关键字和素数P,用移位法定义 ...
- PTA数据结构与算法题目集(中文) 7-42整型关键字的散列映射 (25 分)
PTA数据结构与算法题目集(中文) 7-42整型关键字的散列映射 (25 分) 7-42 整型关键字的散列映射 (25 分) 给定一系列整型关键字和素数P,用除留余数法定义的散列函数将关键字映射 ...
- PTA数据结构与算法题目集(中文) 7-41PAT排名汇总 (25 分)
PTA数据结构与算法题目集(中文) 7-41PAT排名汇总 (25 分) 7-41 PAT排名汇总 (25 分) 计算机程序设计能力考试(Programming Ability Test,简称P ...
- PTA数据结构与算法题目集(中文) 7-40奥运排行榜 (25 分)
PTA数据结构与算法题目集(中文) 7-40奥运排行榜 (25 分) 7-40 奥运排行榜 (25 分) 每年奥运会各大媒体都会公布一个排行榜,但是细心的读者发现,不同国家的排行榜略有不同.比如 ...
- PTA数据结构与算法题目集(中文) 7-39魔法优惠券 (25 分)
PTA数据结构与算法题目集(中文) 7-39魔法优惠券 (25 分) 7-39 魔法优惠券 (25 分) 在火星上有个魔法商店,提供魔法优惠券.每个优惠劵上印有一个整数面值K,表示若你在购买某商 ...
- PTA数据结构与算法题目集(中文) 7-38寻找大富翁 (25 分)
PTA数据结构与算法题目集(中文) 7-38寻找大富翁 (25 分) 7-38 寻找大富翁 (25 分) 胡润研究院的调查显示,截至2017年底,中国个人资产超过1亿元的高净值人群达15万人.假 ...
- PTA数据结构与算法题目集(中文) 7-37 模拟EXCEL排序 (25 分)
PTA数据结构与算法题目集(中文) 7-37 模拟EXCEL排序 (25 分) 7-37 模拟EXCEL排序 (25 分) Excel可以对一组纪录按任意指定列排序.现请编写程序实现类似功能. ...
- PTA数据结构与算法题目集(中文) 7-36 社交网络图中结点的“重要性”计算 (30 分)
PTA数据结构与算法题目集(中文) 7-36 社交网络图中结点的“重要性”计算 (30 分) 7-36 社交网络图中结点的“重要性”计算 (30 分) 在社交网络中,个人或单位(结点)之间通过某 ...
- PTA数据结构与算法题目集(中文) 7-35 城市间紧急救援 (25 分)
PTA数据结构与算法题目集(中文) 7-35 城市间紧急救援 (25 分) 作为一个城市的应急救援队伍的负责人,你有一张特殊的全国地图.在地图上显示有多个分散的城市和一些连接城市的快速道路.每个城市 ...
- PTA数据结构与算法题目集(中文) 7-34
PTA数据结构与算法题目集(中文) 7-34 7-34 任务调度的合理性 (25 分) 假定一个工程项目由一组子任务构成,子任务之间有的可以并行执行,有的必须在完成了其它一些子任务后才能执行.“ ...
随机推荐
- 如何分析SpringBoot源码模块及结构?--SpringBoot源码(二)
注:该源码分析对应SpringBoot版本为2.1.0.RELEASE 1 前言 本篇接 如何搭建自己的SpringBoot源码调试环境?--SpringBoot源码(一). 前面搭建好了自己本地的S ...
- RStudio终端操作
转于:https://support.rstudio.com/hc/en-us/articles/115010737148-Using-the-RStudio-Terminal#send 原文是英文版 ...
- MATLAB神经网络(2) BP神经网络的非线性系统建模——非线性函数拟合
2.1 案例背景 在工程应用中经常会遇到一些复杂的非线性系统,这些系统状态方程复杂,难以用数学方法准确建模.在这种情况下,可以建立BP神经网络表达这些非线性系统.该方法把未知系统看成是一个黑箱,首先用 ...
- HashMap底层源码剖析
HashMap底层源码剖析 一.HashMap底层用到的数据结构 数组+单向链表+红黑树 数组:数组每一项都是一个链表,其实就是数组和链表的结合体 单向链表:当法神hash碰撞时,首先会找到数组对应位 ...
- git 查看commit的提交记录
相关命令: git log 查看所有提交记录 git show 查看提交详情 示例: git log: git show: 查看指定commit的详情:git show commitId 查看某次c ...
- app之---豆果美食
1.抓包 2.代码 抓取: #!/usr/bin/env python # -*- coding: utf-8 -*- #author tom import requests from multipr ...
- dict的常用方法
注:dic表示定义的一个字典变量,如:dic = {'name': 'shawn', 'age': 18} 增: 1. dic['love'] = 'girl' 直接通过新的键值对进行添加 dic ...
- 英伟达GPU虚拟化---申请英伟达测试License
此文基于全新的License 2.0系统,针对vGPU License的试用申请以及软件下载和License管理进行了详细的说明,方便今后我们申请测试License,快速验证GPU的功能. 试用步骤: ...
- CSS导入方式和六种选择器
1.css的导入方式 1.1 行内嵌式 1.2 内部方式 1.2.1含义: css代码写在<head>的<style>标签中 1.2.2 优点 方便在同页面中修改样式 1.2. ...
- masql数据库的表查询
昨日回顾 表与表之间建关系 一对多 换位思考 图书与出版社 先站在左表: 考虑左表的多条数据能否对应右表的一条数据 翻译:多本书能否被一个出版社出版 可以! 注意:单站在一张得出的表关系并不能明确两张 ...