Simpy - 如何异步使用 Filter Store

Simpy - How to use Filter Store asynchronously

我正在使用 Simpy 来模拟机器维修工作。机器可以进厂维修,指派技术人员,然后进行维修,直到完成。

我正在使用 Filter Store 来跟踪技术人员。

NUM_TECHNICIANS = 3               # number of technicians
REPAIR_TIME = 5              # time to repair machine in hours

def task_network(technicians):

    # Request a technician    
    t = yield technicians.get()
    print("Assigned to Technician %d at time %d" % (t.id, env.now))
    yield env.process(t.repair_machine(machine1))

    t = yield technicians.get(lambda t: t.fatigue < 30)
    print("Assigned to Technician %d at time %d" % (t.id, env.now))
    yield env.process(t.repair_machine(machine2))

class Machine(object):
    def __init__(self, env, id, type, isBroken):
        self.env = env
        self.id = id
        self.type = type
        self.isBroken = isBroken

class Technician(object):
    def __init__(self, env, id, skill_level, fatigue, shifts_remaining):
        self.env = env
        self.id = id
        self.skill_level = skill_level
        self.fatigue = fatigue
        self.shifts_remaining = shifts_remaining
        self.isAvailable = True

    def repair_machine(self, machine):
        if machine.type == "MN152":
            self.fatigue += 10
            self.shifts_remaining -= 0.25
            self.isAvailable = False
            print("Repairing...")
            yield env.timeout(REPAIR_TIME)
            print("Technician %d is done repairing at time %d" % (self.id, env.now))

env = simpy.Environment()

# Filter Store allows us to have processes ask for objects as resources (the technicians)
# and get them based off of some criteria (e.g. this radio is complex so requires a more experienced technician)
# If no criteria is specified, Filter Store is FIFO
technicians = simpy.FilterStore(env, capacity = NUM_TECHNICIANS)

t0 = Technician(env, id=0, skill_level=74, fatigue=15, shifts_remaining=2)
t1 = Technician(env, id=1, skill_level=45, fatigue=50, shifts_remaining=1)
t2 = Technician(env, id=2, skill_level=56, fatigue=0, shifts_remaining=3)

technicians.put(jt0)
technicians.put(jt1)
technicians.put(jt2)

machine1 = Machine(env, id=0, type="MN150", isBroken=True)
machine2 = Machine(env, id=1, type="MN152", isBroken=True)

env.process(task_network(technicians))
env.run()

以上代码按预期运行并打印

Assigned to Technician 0 at time 0
Repairing...
Technician 0 is done repairing at time 5
Assigned to Technician 2 at time 5
Repairing...
Technician 2 is done repairing at time 10

但是,如何使机器可以异步进入、分配、维修和 returned?目前(由于我的 yield 语句),模拟暂停直到 yield 进程 returns,这就是为什么它要等到一台机器完成修复后再开始另一台机器修复工作。店里有 3 名技术人员,所以应该允许所有 3 人异步维修 return 机器。我如何实现这一目标?谢谢。

您可以生成多个进程,然后使用 Environment.all_of 等待它们:

procs = [env.process(my_task() for _ in range(3))

# do other stuff ...

results = yield env.all_of(procs)