twisted task.cpperator
twisted task.cpperator
1. twisted task.cpperator
1.1. 简介-cooperator
官方文档:
https://twistedmatrix.com/documents/current/api/twisted.internet.task.Cooperator.html#coiterate
Cooperative task scheduler.
A cooperative task is an iterator where each iteration represents an atomic unit of work. When the iterator yields, it allows the Cooperator to decide which of its tasks to execute next. If the iterator yields a defer.Deferred then work will pause until the defer.Deferred fires and completes its callback chain.
一个协作任务是迭代器,每个迭代元素是一个原子性的操作。当迭代器产出/抛出时,它允许Cooperator决定是否进行下一步。如果迭代器抛出一个Deferred,它会暂停直到Deferred对象fires(触发?)并且跑完了它的回调链。
When a Cooperator has more than one task, it distributes work between all tasks.
如果cooperator对象有多个任务,它会分配执行权到所有任务(简单说就是并发)。
There are two ways to add tasks to a Cooperator, cooperate and coiterate. cooperate is the more useful of the two, as it returns a CooperativeTask, which can be paused, resumed and waited on. coiterate has the same effect, but returns only a defer.Deferred that fires when the task is done.
有两种方式添加任务,cooperate and coiterate,前者功能较多,支持暂停,恢复,等待;后者效用相同,但不能控制执行过程。
Cooperator can be used for many things, including but not limited to:
running one or more computationally intensive tasks without blocking
limiting parallelism by running a subset of the total tasks simultaneously
doing one thing, waiting for a Deferred to fire, doing the next thing, repeat (i.e. serializing a sequence of asynchronous tasks)
Multiple Cooperators do not cooperate with each other, so for most cases you should use the global cooperator.
1.2. 代码解读
前置技能
_EPSILON = 0.00000001
def _defaultScheduler(x):
from twisted.internet
import reactor
return reactor.callLater(_EPSILON,
x)
下文提到的scheduler都是指上述方法,作用是向reactor注册任务
class _Timer(object):
MAX_SLICE = 0.01
def __init__(self):
self.end =
time.time() + self.MAX_SLICE
def __call__(self):
return time.time()
>= self.end
功能:判断延时
主体函数:
class Cooperator(object):
def __init__(self,
terminationPredicateFactory=_Timer,
scheduler=_defaultScheduler,
started=True):
self._tasks = []
self._metarator = iter(())
self._terminationPredicateFactory = terminationPredicateFactory
self._scheduler = scheduler
self._delayedCall = None
self._stopped = False
self._started = started def coiterate(self, iterator, doneDeferred=None):
if doneDeferred is None:
doneDeferred = defer.Deferred()
CooperativeTask(iterator, self).whenDone().chainDeferred(doneDeferred)
return doneDeferred def cooperate(self, iterator):
return CooperativeTask(iterator, self)
最重要的两个方法。
1.2.1. cooperate
先看一下def cooperate(self, iterator):
class CooperativeTask(object):
def __init__(self,
iterator, cooperator):
"""
A private constructor: to create
a new L{CooperativeTask}, see
L{Cooperator.cooperate}.
"""
self._iterator
= iterator
self._cooperator
= cooperator
self._deferreds
= []
self._pauseCount
= 0
self._completionState
= None
self._completionResult
= None
cooperator._addTask(self)
回到cooperator._addTask()
def _addTask(self, task):
"""
Add a L{CooperativeTask} object to
this L{Cooperator}.
"""
if self._stopped:
self._tasks.append(task) # XXX silly, I know, but _completeWith
# does the inverse
task._completeWith(SchedulerStopped(),
Failure(SchedulerStopped()))
else:
self._tasks.append(task)
self._reschedule()
_mustScheduleOnStart = False
def _reschedule(self):
if not self._started:
self._mustScheduleOnStart = True
return
if self._delayedCall is None
and self._tasks:
self._delayedCall =
self._scheduler(self._tick)
通过self._scheduler(self._tick)注册了一个调用,实例化到这里就结束了。
这也是twisted的一个特点,经常是向循环注册一个方法,后续执行从循环开始。
待reactor开始之后,会调用_tick
def _tasksWhileNotStopped(self):
"""
Yield all L{CooperativeTask} objects
in a loop as long as this
L{Cooperator}'s termination condition
has not been met.
"""
terminator =
self._terminationPredicateFactory()
while self._tasks:
for t in self._metarator:
yield t
if terminator():
return
self._metarator = iter(self._tasks)
def _tick(self):
"""
Run one scheduler tick.
"""
self._delayedCall = None
for taskObj in self._tasksWhileNotStopped():
taskObj._oneWorkUnit()
self._reschedule()
terminator()是用于执行时间片长度的判断,用于保证在多个task之间按时间片切换,定制时间片长可以重写_timer并在创建cooperator时指定。
返回的t实际是一个class CooperativeTask(object)实例
def _oneWorkUnit(self):
"""
Perform one unit of work for this
task, retrieving one item from its
iterator, stopping if there are no
further items in the iterator, and
pausing if the result was a
L{defer.Deferred}.
"""
try:
result = next(self._iterator)
except StopIteration:
self._completeWith(TaskDone(),
self._iterator)
except:
self._completeWith(TaskFailed(),
Failure())
else:
if isinstance(result,
defer.Deferred):
self.pause()
def failLater(f):
self._completeWith(TaskFailed(),
f)
result.addCallbacks(lambda result:
self.resume(),
failLater)
result = next(self._iterator)
这里就是传入的iter对象了
if isinstance(result, defer.Deferred):
注意,如果result不是延迟调用,则直接执行,然后循环;
如果它是deferred,执行self.pause(),在其中会把self从_task队列中删除。
同时为result添加callback,在回调中执行self.resume()
def resume(self):
"""
Resume processing of a paused
L{CooperativeTask}.
@raise NotPaused: if this
L{CooperativeTask} is not paused.
"""
if self._pauseCount
== 0:
raise NotPaused()
self._pauseCount -= 1
if self._pauseCount
== 0 and self._completionState is None:
self._cooperator._addTask(self)
在resume中重新执行_addTask添加任务,注册tick调用。
通过上述过程决定了一个deferred只能在执行-暂停-重新注册任务-向reactor注册任务-调用-执行的过程中循环,也就是执行流程只能一一执行。
1.2.2.
coiterate
coiterate解读
def coiterate(self,
iterator, doneDeferred=None):
if doneDeferred is None:
doneDeferred =
defer.Deferred()
CooperativeTask(iterator,
self).whenDone().chainDeferred(doneDeferred)
return doneDeferred
与cooperate()不同的是,它返回一个deferred对象。
def whenDone(self):
d = defer.Deferred()
if self._completionState
is None:
self._deferreds.append(d)
else:
d.callback(self._completionResult)
return d
整个过程实质就是在CooperativeTask的_deferreds列表中保存deferred对象,并为这个对象添加一个回调指向doneDeferred的回调,然后返回doneDeferred
结果就是当iterator结束或异常时都会调用CooperativeTask的_completeWith
def _completeWith(self, completionState,
deferredResult):
self._completionState
= completionState
self._completionResult =
deferredResult
if not self._pauseCount:
self._cooperator._removeTask(self)
for d in self._deferreds:
d.callback(deferredResult)
调用_completeWith就标志着迭代结束或异常,做了两个操作,清除task,调用self._deferreds中每个对象的回调。
这一操作会触发doneDeferred的回调;至此所有处理流程结束,cooperator执行操作,返回延迟调用对象,触发回调。
twisted task.cpperator的更多相关文章
- twisted的task之cooperator和scrapy的parallel()函数
def handle_spider_output(self, result, request, response, spider): if not result: return defer_succe ...
- Python自动化运维之15、网络编程之socket、socketserver、select、twisted
一.TCP/IP相关知识 TCP/UDP提供进程地址,两个协议互不干扰的独自的协议 TCP :Transmission Control Protocol 传输控制协议,面向连接的协议,通信 ...
- twisted(1)--何为异步
早就想写一篇文章,整体介绍python的2个异步库,twisted和tornado.我们在开发python的tcpserver时候,通常只会用3个库,twisted.tornado和gevent,其中 ...
- twisted(3)--再谈twisted
上一章,我们直接写了一个小例子来从整体讲述twisted运行的大致过程,今天我们首先深入一些概念,在逐渐明白这些概念以后,我们会修改昨天写的例子. 先看下面一张图: 这个系列的第一篇文章,我们已经为大 ...
- twisted(2)--聊天系统
我们今天要做一个聊天系统,这样可以和我们之前flask api那系列文章结合起来:其次,聊天系统最能代表tcpserver,以后可以套用各种模型,比如我们公司做的物联网,其实就是把聊天系统简化一下. ...
- twisted的一些代码
之前用swoole(1.7.19)写的一段程序在数据量大的时候存在内存泄漏,改为twisted(15.4)实现,自测无误,记录如下(两者cpu占用率90%时吞吐rps能从120提升到1000). #! ...
- twisted学习笔记 No.2 WebServer
原创博文,转载请注明出处. 当服务器接收到一个客户端请求后,会创建一个请求对象并传递到资源系统,资源系统会根据请求路径分发到相应的资源对象,资源被要求渲染自身并返回结果到客户端. 解析HTTP Req ...
- 理解Twisted与非阻塞编程
先来看一段代码: # ~*~ Twisted - A Python tale ~*~ from time import sleep # Hello, I'm a developer and I mai ...
- 转载 twisted(1)--何为异步
Reference: http://www.cnblogs.com/yueerwanwan0204/p/5589860.html 早就想写一篇文章,整体介绍python的2个异步库,twisted和t ...
随机推荐
- 防止不同账号之间localStorage数据错误
set和get的时候,key后面加上用户ID
- Educational Codeforces Round 76 (Rated for Div. 2) C. Dominated Subarray
Let's call an array tt dominated by value vv in the next situation. At first, array tt should have a ...
- 程序设计实验:一个Python游戏,体验软件开发。
小组在GitHub上找了一个Pygame实现的超级马里奥游戏.所以我的学习过程大致如下: 1.快速学习Python基础语法. 2.学习pygame并着手理解这个项目. 3.完成作业以及各种文档报告. ...
- MySql 怎么存取 Emoji
01.前言 Emoji 在我们生活中真的是越来越常见了,几乎每次发消息的时候不带个 Emoji,总觉得少了点什么,似乎干巴巴的文字已经无法承载我们丰富的感情了.对于我们开发者来说,如何将 Emoji ...
- docker容器 - 新建容器、启动容器、暂停容器和停止容器
实验环境 CentOS 7.5 容器 容器是镜像的运行实例.不同的是,镜像是静态的只读文件,而容器带有运行时需要的可写文件层:同时,容器中的应用进程处于运行状态. 新建容器 [root@kvm ~]# ...
- 对C#继承、多态的理解
11月3日 阴天 前两天看某位大牛写的程序,对于C#多态有困惑,今天一大早来查阅了不少资料,自认为有了一个基本的认知,记录下来,一扫今天这阴霾的天气 ------------------------- ...
- C语言程序设计100例之(27):回旋方阵
例27 回旋方阵 问题描述 编写程序,生成从内到外是连续的自然数排列的回旋方阵.例如,当n=3和n=4时的回旋方阵如下图1所示. 图1 由内到外回旋方阵 输入格式 一个正整数n(1≤n ...
- web前端-基础篇
该篇仅是本人学习前端时,做的备忘笔记: 一.背景图片设置: 设置背景图时的css代码:background-image:url(图片的url路径); ps:设置好这个背景后请一定要设置该背景图片的大小 ...
- POJ 3991 括号匹配问题(贪心)
I’m out of stories. For years I’ve been writing stories, some rather silly, just to make simple prob ...
- gulp 搭建静态服务器
步骤: 安装依赖:npm i browser-sync --save-dev 导入browser-sync,通过create创建 设置Sass和Js任务,将其压缩重命名并引入页面,任务结束时reloa ...