python开发实例之Python的Twisted框架中Deferred对象的详细用法与实例


Posted in Python onMarch 19, 2020

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): : The headline ``123456789012345678901234567890123456789012345678901234567890'' is too long!]

python开发实例之Python的Twisted框架中Deferred对象的详细用法与实例

我们用图看一下触发流程:

  • 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的Twisted框架中Deferred对象的详细用法,更多关于Python的Twisted框架知识技巧请查看下面的相关链接

Python 相关文章推荐
Python批量重命名同一文件夹下文件的方法
May 25 Python
python 读写txt文件 json文件的实现方法
Oct 22 Python
Windows下Python3.6安装第三方模块的方法
Nov 22 Python
详解python中init方法和随机数方法
Mar 13 Python
Python 通过打码平台实现验证码的实现
May 13 Python
python读取图片的方式,以及将图片以三维数组的形式输出方法
Jul 03 Python
python 3.7.4 安装 opencv的教程
Oct 10 Python
解决pandas展示数据输出时列名不能对齐的问题
Nov 18 Python
解决Keras使用GPU资源耗尽的问题
Jun 22 Python
Python 忽略文件名编码的方法
Aug 01 Python
python实现单机五子棋
Aug 28 Python
Python如何截图保存的三种方法(小结)
Sep 01 Python
mac 上配置Pycharm连接远程服务器并实现使用远程服务器Python解释器的方法
Mar 19 #Python
python数据库开发之MongoDB安装及Python3操作MongoDB数据库详细方法与实例
Mar 18 #Python
Python3开发实例之非关系型图数据库Neo4j安装方法及Python3连接操作Neo4j方法实例
Mar 18 #Python
selenium+python配置chrome浏览器的选项的实现
Mar 18 #Python
python开发实例之python使用Websocket库开发简单聊天工具实例详解(python+Websocket+JS)
Mar 18 #Python
python selenium操作cookie的实现
Mar 18 #Python
Selenium 滚动页面至元素可见的方法
Mar 18 #Python
You might like
ThinkPHP中url隐藏入口文件后接收alipay传值的方法
2014/12/09 PHP
在Mac OS上自行编译安装Apache服务器和PHP解释器
2015/12/24 PHP
详解no input file specified 三种解决方法
2019/11/29 PHP
基于Jquery的文字自动截取(提供源代码)
2011/08/09 Javascript
JS Pro-深入面向对象的程序设计之继承的详解
2013/05/07 Javascript
动态加载脚本提升javascript性能
2014/02/24 Javascript
jQuery filter函数使用方法
2014/05/19 Javascript
jquery html动态添加的元素绑定事件详解
2016/05/24 Javascript
JavaScript程序中实现继承特性的方式总结
2016/06/24 Javascript
解析Vue.js中的组件
2018/02/02 Javascript
JS实现的JSON数组去重算法示例
2018/04/11 Javascript
vux uploader 图片上传组件的安装使用方法
2018/05/15 Javascript
对vuejs的v-for遍历、v-bind动态改变值、v-if进行判断的实例讲解
2018/08/27 Javascript
mpvue实现小程序签到金币掉落动画(api实现)
2019/10/17 Javascript
[47:21]Liquid vs TNC Supermajor 胜者组 BO3 第一场 6.4
2018/06/05 DOTA
Python搭建APNS苹果推送通知推送服务的相关模块使用指南
2016/06/02 Python
Windows 8.1 64bit下搭建 Scrapy 0.22 环境
2018/11/18 Python
解决pandas展示数据输出时列名不能对齐的问题
2019/11/18 Python
Pytorch 实现数据集自定义读取
2020/01/18 Python
python中np是做什么的
2020/07/21 Python
什么是Python包的循环导入
2020/09/08 Python
HTML5中的新元素介绍
2008/10/17 HTML / CSS
什么造成了Java里面的异常
2016/04/24 面试题
计算机专业应届毕业生自荐信
2013/09/26 职场文书
幼儿园新学期寄语
2014/01/18 职场文书
项目建议书怎么写
2014/05/15 职场文书
材料专业大学毕业生自荐书
2014/07/02 职场文书
教师批评与自我批评发言稿
2014/10/15 职场文书
党员考试作弊检讨书1000字
2015/02/16 职场文书
酒店总经理岗位职责
2015/04/01 职场文书
pandas DataFrame.shift()函数的具体使用
2021/05/24 Python
PyMongo 查询数据的实现
2021/06/28 Python
教你利用Nginx 服务搭建子域环境提升二维地图加载性能的步骤
2021/09/25 Servers
mysql使用instr达到in(字符串)的效果
2022/04/03 MySQL
Nginx如何配置多个服务域名解析共用80端口详解
2022/09/23 Servers
python中validators库的使用方法详解
2022/09/23 Python