2012-07-15 73 views
2

參數使我在使用延遲函數時扭曲了很多。參數在使用扭曲時在循環中延遲函數

我認爲如果函數的參數是一個INT值VALUE,或者除了list和dict以外的其他基本類型,它應該傳遞值而不是引用。

from twisted.internet import defer, reactor 

def deferFunc(x): 
    print "11111 %d" % x 
    d = defer.Deferred() 
    reactor.callLater(1, d.callback, x) 
    return d 

@defer.inlineCallbacks 
def inlineDeferFunc(x): 
    print "11111 %d" % x 
    d = defer.Deferred() 
    reactor.callLater(1, d.callback, x) 
    result = yield d 
    defer.returnValue(result) 

def loop(): 
    x = [1,2,3,4] 
    d = defer.succeed(0) 
    for i in x: 
     d.addCallback(lambda _ : inlineDeferFunc(i)) 

if __name__ == '__main__': 
    loop() 

    try: 
     reactor.run() 
    except: 
     reactor.stop() 

結果是:

11111 1 
11111 4 
11111 4 
11111 4 

,結果搞糊塗了。

我怎樣才能像1,2,3,4

+0

通過我發現 'd.addCallback(拉姆達_:inlineDeferFunc(X [0]))的方式\ \t .addCallback(拉姆達_:inlineDeferFunc(X [1]))\ \t。 addCallback(lambda _:inlineDeferFunc(x [2]))\ \t .addCallback(lambda _:inlineDeferFunc(x [3]))\' 這將工作正常。所以我認爲這與** for循環有關。 – holsety 2012-07-15 02:38:28

回答

3

正確的輸出我怎樣才能像1,2,3,4

這樣正確的輸出:

def loop(): 
    x = [1,2,3,4] 
    d = defer.succeed(0) 
    for i in x: 
     d.addCallback(lambda _, i = i : inlineDeferFunc(i)) 



11111 1 
11111 2 
11111 3 
11111 4 

請注意,我們正在複製函數定義i = i中的值。
的問題是,lambda抓住家長環境來訪問變量i畢竟它是如何知道哪些價值,對於性能方面的原因,我不認爲它複製它的定義中,但父環境不斷被更新for循環。

這裏是一個簡單的測試。

>>> def test_closures(): 
...  i = 2 
...  def test(): 
...   return i 
...  i = 3 
...  return test 
... 
>>> test_closures()() 
3 
+0

謝謝!它必須是** lambda **問題。 – holsety 2012-07-15 03:01:36