admin管理员组文章数量:1352026
Some context about the original simulation. I have 2 processes P1 and P2 which are based off the same function. This function looks through FilterStore.items first to "screen" for an item with the lowest value of a certain attribute, and then performs a get with the exact value of that attribute which will get a desired item. If the store was empty, then it will simply have a parameterless get which yields the first item put into the store. If you need an analogy, one could be consumers wanting to buy a GPU with the lowest cost, but if there was no GPU in stock, they would just get their hands on whatever GPU becomes available first.
Now, suppose the store was empty at first, so when P1 executes, it yields for a parameterless get. Subsequently, store.put(item)
and then P2 is in the event queue at the same time in that order. At the end, P1 gets the item ... but somehow P2 is yielding for a filtered get instead of a parameterless one!
Here is a sample code, and I have tagged the env._queue
print statements with a number before it so that I can reference to the output and the order as well. Note that foo()
and bar()
are actually the same functions, but represent P1 and P2. baz()
is a method to perform put
.
import simpy
class something:
def __init__(self, dist):
self.dist = dist
env = simpy.Environment()
store = simpy.FilterStore(env)
def foo():
min_dist = float('inf')
for item in store.items:
if item.dist < min_dist:
min_dist = item.dist
if min_dist != float('inf'):
# Will never execute for this example
item = yield store.get(lambda item: item.dist == min_dist)
else:
item = store.get()
item.callbacks.append(lambda _: print("7) foo get has completed " + str(env._queue)))
print(f"foo is now in the else-block, foo's get process is {item}")
print(f'1) {env._queue}')
item = yield item
print(f"8) {env._queue}") # Foo got the item!
def bar():
min_dist = float('inf')
for item in store.items:
if item.dist < min_dist:
min_dist = item.dist
print(f'4) {env._queue}')
if min_dist != float('inf'):
item = store.get(lambda item: item.dist == min_dist)
print(f"bar is now in the if-block, bar's get process is {item}")
print(f'5) {env._queue}')
item.callbacks.append(lambda _: print("bar get complete " + str(env._queue)))
item = yield item
print(f"bar gets the item, queue = {env._queue}")
else:
# This was the branch I expected to execute, but it did not
item = yield store.get()
def baz():
print(f'2) {env._queue}')
print(f"baz has not put in the object, store items = {store.items}")
put = store.put(something(2))
put.callbacks.append(lambda _: print(f"6) baz first put has completed" + str(env._queue)))
print(f"baz has put in the object, store items = {store.items}")
print(f'3) {env._queue}')
yield env.timeout(50)
print(f'9) {env._queue}, this is after the env.timeout. At this point, bar is still waiting')
put = store.put(something(2))
print(f"10) {env._queue}, final put is {put}")
put.callbacks.append(lambda _: print(f"11) final put {put} has complete " + str(env._queue)))
env.process(foo())
env.process(baz())
env.process(bar())
print(f'0) {env._queue}')
env.run()
Some questions I have regarding the output, which can be gotten by running the sample code.
In statement 3 and the print statement above it, a
StorePut()
was triggered and placed after theInitialize()
in the event queue. However, the item is already in the store before the actualStorePut()
has been processed. What purpose doesStorePut()
then do? Does it notify the store to trigger any processes waiting for an item?In statement 5 after
bar
has entered the wrong branch and performed a filteredget
. At this point, aFilterStoreGet() object
belonging tofoo
is triggered and placed in the queue. Can I think of thisget
bybar
as additionally notifying the store to perform a check and triggers the original request byfoo
as an item is now in the store?In statement 11 (within the callback),
baz
should have terminated, but how come there is aProcess(baz) object
in the queue, especially since theput
being processed should imply thatbaz
has been processed? Notice that all other callback statements do not have it.Finally, with regards to the first question, is there a way for me to modify
put
behaviour, so that the item is NOT in the stall until the event is processed or if that is not possible, is there a way to "delay" this so thatbar
executes before the firstput
event? With this, P2 orbar
can go down the intended branch.
Thank you all in advance for any insight that is provided!
本文标签:
版权声明:本文标题:Trying to understand event schedulingenv._queue with regards to SimPy's FilterStore put and get events - Stack Overflow 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1743907080a2559718.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论