问题描述
我正在使用 simpy 来模拟制造设备。设备需要使用一些指定的材料来制造产品。我尝试使用 FilterStore 对此进行建模,如下面的代码所示。
import simpy
class Material:
def __init__(self,name):
self.name = name
class Machine(object):
def __init__(self,env):
self.env = env
self.inputs = simpy.FilterStore(env)
def run(self):
mat = [ self.inputs.get(lambda i: i.name == itemname ) for itemname in ["mat1","mat2","mat4"] ]
res = yield self.env.all_of(mat)
print([res.events[i]._value.name for i in range(len(res.events))])
def input_materials(env,m):
for i in range(5):
m.inputs.put(Material( "mat"+str(i) ))
m.inputs.put(Material( "mat"+str(i) ))
m.inputs.put(Material( "mat"+str(i) ))
yield env.timeout(1)
env = simpy.Environment()
machine = Machine(env)
env.process(machine.run())
env.process(input_materials(env,machine))
env.run()
['mat4','mat4','mat4']
我想得到[mat1,mat2,mat4],但是上面的代码都得到了mat4。 我们已经确认,如果我不使用变量,项目名称,并按如下方式单独编码,则结果符合预期。
def run(self):
m1 = self.inputs.get(lambda i: i.name == "mat1" )
m2 = self.inputs.get(lambda i: i.name == "mat2" )
m4 = self.inputs.get(lambda i: i.name == "mat4" )
res = yield self.env.all_of([m1,m2,m4])
print([res.events[i]._value.name for i in range(len(res.events))])
我应该如何编码?一些帮助将不胜感激。
解决方法
这很微妙。
lambda 中的 itemname 与 for
语句中使用的 itemname 相同
但是,直到 for
语句完成后,lambda 才会执行,
这意味着当 lambdas 最终运行时,它们都使用来自 for
语句的相同 itemname 变量以及 for
语句的最后一个 mat4 循环赋值。
这也被称为一个非常有用的外壳
要解决此问题,您的 lambda 需要一个本地 itemname 变量
更新
mat = [ self.inputs.get(lambda i: i.name == itemname ) for itemname in ["mat1","mat2","mat4"] ]
到
mat = [ self.inputs.get(lambda i,itemname=itemname: i.name == itemname ) for itemname in ["mat1","mat4"] ]