實例解析Python的Twisted框架中Deferred對象的用法

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中的關鍵之處

  1. Deferreds將會在調用其callback或errback時被觸發;
  2. Deferreds僅能被觸發一次!如果嘗試多次觸發將會導致AlreadyCalledError異常;
  3. 第N級callback或errback中的Exceptions將會傳入第N+1級的errback中;如果沒有errback,則會拋出Unhandled Error。如果第N級callback或errback中沒有拋出Exception或返回Failure對象,那接下來將會由第N+1級中的callback進行處理;
  4. callback中返回的結果將會傳入下一級callback,並作爲其第一個參數;
  5. 如果傳入errback的錯誤不是一個Failure對象,那將會被自動包裝一次。
    最後給大家推薦一個口碑不錯的python聚集地【點擊進入】,這裏有很多的老前輩學習技巧,學習心得

,面試技巧,職場經歷等分享,更爲大家精心準備了零基礎入門資料,實戰項目資料,每天都有程序員

定時講解Python技術,分享一些學習的方法和需要留意的小細節

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章