提问人:Dan 提问时间:9/21/2023 更新时间:9/22/2023 访问量:49
拓扑排序?不使用 Topsort
Topological sort? without using topsort
问:
我有一个名为 activities.py 的 python 列表
activities = [{'Cost': 10000, 'Duration': 5, 'Name': 'Activity 1', 'Predecessors': None},
{'Cost': 8000, 'Duration': 3, 'Name': 'Activity 2', 'Predecessors': ['Activity 1']},
{'Cost': 2000, 'Duration': 15, 'Name': 'Activity 3', 'Predecessors': ['Activity 2', 'Activity 1']},
{'Cost': 7000, 'Duration': 16, 'Name': 'Activity 4', 'Predecessors': ['Activity 2']},
{'Cost': 5000, 'Duration': 20, 'Name': 'Activity 5', 'Predecessors': ['Activity 4']},
...
该列表表示一个由具有设定持续时间的活动组成的项目,并且还具有“前置任务”键,该键指示在开始该活动之前必须完成哪些活动。该列表一直延伸到活动 1001,但我只包含了前 5 行。
我已经使用以下方法计算了所有活动的总时间:
def get_total_project_duration(activities: list) -> int:
total_time = sum(item['Duration'] for item in activities)
return total_time
pass
但是,我想找出尽可能短的“持续时间”,因为该列表有一个“前置任务”键,它允许多个活动同时启动,以尽可能少地花费时间。 我知道这可以使用拓扑排序来完成,但我想看看是否有办法仅使用默认库来做同样的事情。
我已经尝试过使用图表,但数据集太大了,无法解决
答:
0赞
tobias_k
9/21/2023
#1
您可以编写一个缓存的递归函数(例如使用 )来确定每个活动的最早可能结束时间,递归到前一个活动(如果有的话),并获取以下函数:functools.cache
max
from functools import cache
act_dict = {a["Name"]: a for a in activities}
@cache
def earliest_finish(name):
pred = max(map(earliest_finish, act_dict[name]["Predecessors"] or []), default=0)
return pred + act_dict[name]["Duration"]
res = {a: earliest_finish(a) for a in act_dict}
print(res)
print(max(res.values()))
显示的五个活动的输出:
{'Activity 1': 5, 'Activity 2': 8, 'Activity 3': 23, 'Activity 4': 24, 'Activity 5': 44}
44
无论活动在列表中的显示顺序如何,这都有效。根据需要计算前置任务,然后缓存。
0赞
Alain T.
9/22/2023
#2
我建议将您的活动存储在字典中,并以它们的名称为键。然后,您可以遍历这些活动,根据先前计算的“完成”值向每个活动添加一个“完成”值(使用 FIFO 队列推迟具有不完整前置任务的活动)
输入:
activities = [{'Cost': 10000, 'Duration': 5, 'Name': 'Activity 1', 'Predecessors': None},
{'Cost': 8000, 'Duration': 3, 'Name': 'Activity 2', 'Predecessors': ['Activity 1']},
{'Cost': 2000, 'Duration': 15, 'Name': 'Activity 3', 'Predecessors': ['Activity 2', 'Activity 1']},
{'Cost': 7000, 'Duration': 16, 'Name': 'Activity 4', 'Predecessors': ['Activity 2']},
{'Cost': 5000, 'Duration': 20, 'Name': 'Activity 5', 'Predecessors': ['Activity 4']} ]
过程:
from collections import deque
actDict = { act['Name']:act for act in activities }
links = deque(actDict.values())
while links:
act = links.popleft()
preds = [ pred.get('Finish') for pred in
map(actDict.get,act.get('Predecessors') or []) ]
if None in preds:
links.append(act)
else:
act['Finish'] = act['Duration'] + max(preds,default=0)
输出:
print(actDict)
{'Activity 1':
{'Cost': 10000, 'Duration': 5, 'Name': 'Activity 1',
'Predecessors': None, 'Finish': 5},
'Activity 2':
{'Cost': 8000, 'Duration': 3, 'Name': 'Activity 2',
'Predecessors': ['Activity 1'], 'Finish': 8},
'Activity 3':
{'Cost': 2000, 'Duration': 15, 'Name': 'Activity 3',
'Predecessors': ['Activity 2', 'Activity 1'], 'Finish': 23},
'Activity 4':
{'Cost': 7000, 'Duration': 16, 'Name': 'Activity 4',
'Predecessors': ['Activity 2'], 'Finish': 24},
'Activity 5':
{'Cost': 5000, 'Duration': 20, 'Name': 'Activity 5',
'Predecessors': ['Activity 4'], 'Finish': 44}}
评论