拓扑排序?不使用 Topsort

Topological sort? without using topsort

提问人:Dan 提问时间:9/21/2023 更新时间:9/22/2023 访问量:49

问:

我有一个名为 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

但是,我想找出尽可能短的“持续时间”,因为该列表有一个“前置任务”键,它允许多个活动同时启动,以尽可能少地花费时间。 我知道这可以使用拓扑排序来完成,但我想看看是否有办法仅使用默认库来做同样的事情。

我已经尝试过使用图表,但数据集太大了,无法解决

Python 列表 排序

评论


答:

0赞 tobias_k 9/21/2023 #1

您可以编写一个缓存的递归函数(例如使用 )来确定每个活动的最早可能结束时间,递归到前一个活动(如果有的话),并获取以下函数:functools.cachemax

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}}