Deferred對象在Twsited框架中用於處理回調,這對於依靠異步的Twisted來說十分重要,接下來我們就以實例解析Python的Twisted框架中Deferred對象的用法
Deferred對象結構
Deferred由一系列成對的回調鏈組成,每一對都包含一個用於處理成功的回調(callbacks)和一個用於處理錯誤的回調(errbacks)。初始狀態下,deffereds將由兩個空回調鏈組成。在向其中添加回調時將總是成對添加。當異步處理中的結果返回時,Deferred將會啓動並以添加時的順序觸發回調鏈。
用實例也許更容易說明,首先來看看addCallback:
from twisted.internet.defer import Deferred
def myCallback(result):
print result
d = Deferred()
d.addCallback(myCallback)
d.callback("Triggering callback.")
運行它將會得到如下結果:
Triggering callback.
上例中創建了一個deffered並利用其addCallback方法註冊一個用於處理成功的回調。d.callback會啓動deffered並調用callback鏈。傳入callback的參數也會被各callback鏈中的第一個函數接收到。
有addCallback,那另一個錯誤的分支,我想也能猜測到了那就是addErrorback,同樣來看個例子:
from twisted.internet.defer import Deferred
def myErrback(failure):
print failure
d = Deferred()
d.addErrback(myErrback)
d.errback(ValueError("Triggering errback."))
運行它將會得到如下結果:
[Failure instance: Traceback (failure with no frames): <type 'exceptions.ValueError'>: Triggering errback.]
可以看出Twisted會把錯誤封裝在Failure裏。
值得注意的是,在之前提到過註冊回調總是成對的。在使用d.addCallback和d.addErrorback方法時,我們看似只是添加了一個callback或一個errback。而實際上,爲了完成這一級回調鏈的創建,這些方法還會爲另一半註冊一個pass-through。要記住,回調鏈總是具有相同的長度。如果要分別指定這一級回調的callback和errback。可以使用d.addCallbacks方法:
d = Deferred()
d.addCallbacks(myCallback, myErrback)
d.callback("Triggering callback.")
那麼…今天就先到這裏。
進階示例
接下來就應該來點更爲實際的,那就是放進Reactor。先來看一個例子:
from twisted.internet import reactor, defer
class HeadlineRetriever(object):
def processHeadline(self, headline):
if len(headline) > 50:
self.d.errback(Exception("The headline ``%s'' is too long!" % (headline,)))
else:
self.d.callback(headline)
def _toHTML(self, result):
return "<h1>%s</h1>" % (result,)
def getHeadline(self, input):
self.d = defer.Deferred()
reactor.callLater(1, self.processHeadline, input)
self.d.addCallback(self._toHTML)
return self.d
def printData(result):
print result
reactor.stop()
def printError(failure):
print failure
reactor.stop()
h = HeadlineRetriever()
d = h.getHeadline("Breaking News: Twisted Takes us to the Moon!")
d.addCallbacks(printData, printError)
reactor.run()
上例接收一個標題並對其進行處理,如果標題超長會返回超長的錯誤,否則將其轉爲HTML並返回。
因所給的標題少於50個字符,故執行以上代碼會得到如下返回:
<h1>Breaking News: Twisted Takes us to the Moon!</h1>
有一點值得注意的,上面用到了reactor的callLater方法,它可以用來做定時事件從而模擬一個異步的請求。
如果我們將標題變得很長,比如說:
h = HeadlineRetriever()
d = h.getHeadline("1234567890"*6)
d.addCallbacks(printData, printError)
那結果是可以遇見的
[Failure instance: Traceback (failure with no frames): <type 'exceptions.Exception'>: The headline ``123456789012345678901234567890123456789012345678901234567890'' is too long!]
我們用圖看一下觸發流程:
Deferreds中的關鍵之處
- Deferreds將會在調用其callback或errback時被觸發;
- Deferreds僅能被觸發一次!如果嘗試多次觸發將會導致AlreadyCalledError異常;
- 第N級callback或errback中的Exceptions將會傳入第N+1級的errback中;如果沒有errback,則會拋出Unhandled Error。如果第N級callback或errback中沒有拋出Exception或返回Failure對象,那接下來將會由第N+1級中的callback進行處理;
- callback中返回的結果將會傳入下一級callback,並作爲其第一個參數;
- 如果傳入errback的錯誤不是一個Failure對象,那將會被自動包裝一次。
最後給大家推薦一個口碑不錯的python聚集地【點擊進入】,這裏有很多的老前輩學習技巧,學習心得
,面試技巧,職場經歷等分享,更爲大家精心準備了零基礎入門資料,實戰項目資料,每天都有程序員
定時講解Python技術,分享一些學習的方法和需要留意的小細節