Python中reactor,factory,protocol

最为简单的情况下,除了了解清reactor的简单使用,你还要了解Protocol和Factory。它们最终都会由reactor的侦听建立和run来统一调度起来。

 建立服务器的第一个要解决的问题就是服务与外界的交流协议。协议的定义在twisted中是通过继承twisted.internet.protocol.Protocol类来实现的。在协议中你可以定义连接、数据接收、断开连接等一系列的事件如果进行响应。但是对于所有连接上来的协议的建立、持久配置数据的存储这样的内容并不会保存在协议中。     
    
   持久配置数据的存储通常都会保存在工厂里。     
    
     工厂的定义在twisted中是通过继承twisted.internet.protocol.Factory类来实现的。twisted提供了缺省的工厂实现最为普通的需求。它会实例化每个协议,并且通过设置每个协议中factory属性来使协议可以使用到它自己,做这个设置的作用就是让协议在进行处理连接数据时可以使用到工厂中存储的持久配置数据。工厂的启动是需要reactor真实的建立侦听并启动才可以实现的。     
 reactor侦听的建立可以参考   twisted.internet.interfaces.IReactorTCP.listenTCP。这时我们先只说建立TCP服务器的侦听,如果你需要其它种类的侦听建立参考IReactor*.listen*系列API。     
    
  总结一下,我们书写一个twisted的Daemon,实质上会关注三个层次的对象。它们互相可配置,可独立开发,只需要通过简单的调用配置就可结合使用。第一层次就是侦听的建立、工厂的初始化、服务器的运行,它需要reactor的开发。第二个层次就是服务的初始化、用户连接的建立、持久配置数据的存储、协议的实例化,它需要factory的开发。第三个层次就是用户连接建立后的事件处理,这就需要protocol的开发了。

一、

protocol:内部实现的主要是连接在通信时的动作;

内部有transport,可进行write(), getpeer()(host,port)

还有factory

Factory:保存的是连接方的信息,当有链接过来时,factory会初始化一个protocol与对方建立连接,所以factory中有protocol.  因为protocol的方法经常会用到factory里的属性变量,所以protocol类中也有factory。这里就有一个循环引用的问题,Python中,有些类是有垃圾清理机制的,但是对于那些有定义方法__del__()的类,就会有内存泄露的问题。

Reactor:是管理twisted框架的核心。所有的事件都会触发reactor,然后他会开启服务,初始化factory,factory再初始化protocol。

Factory和protocol的基础实现在protocol.py文件中。

二、

Reactor:

是twisted框架中很重要的概念,事件驱动。他负责监测所有事件。

最常用的是run(), stop(), callLater()

callLater()实际是实例化了一个DelayedCall()类,并将实例的句柄返回。可进行cancel(), reset(), delay()等操作。

Reactor的实现,应该继承了internet\base.py文件中的类ReactorBase(object)还有类_SignalReactorMixin,其中,_SignalReactorMixin类中有定义run()函数。

ReactorBase(object)类中,有方法stop()。好多reactor和thread的操作函数也能在这里面找到,上面的callLater()就是在这里实现的。

在internet\posixbase.py文件中,有类PosixReactorBase(_SignalReactorMixin, ReactorBase),是上面说到的两个类的继承。在这里,实现了各种listen**函数,主要服务员server;还有对应的connect**函数,针对client。

<一>、例如:客户端可以使用:reactor. connectTCP(self, host, port, factory, timeout=30, bindAddress=None)

这是一个直接可以用的,所以reactor必定继承自PosixReactorBase。

在connectTCP()中,实现如下:

c = tcp.Connector(host, port, factory, timeout, bindAddress, self)

c.connect()

return c

第一句:实例一个tcp.Connector类,该类继承自base. BaseConnector;

类base. BaseConnector实现的方法有:disconnect(), connect(),stopConnecting(self), cancelTimeout(self), buildProtocol(self, addr), connectionFailed(self, reason), connectionLost(self, reason)…

这些方法,和factory中的方法,函数名很相似,其实这些函数的内部就是调用相应的factory方法的。该类中有个很重要的变量就是self.factory。

第二句:是连接的发起函数,主要是开启了factory(connectTCP函数中传入的)。

Connect()定义如下:

def connect(self):

"""Start connection to remote server."""

if self.state != "disconnected":

raise RuntimeError, "can‘t connect in this state"

self.state = "connecting"

if not self.factoryStarted:

self.factory.doStart()

self.factoryStarted = 1

self.transport = transport = self._makeTransport()

if self.timeout is not None:

self.timeoutID = self.reactor.callLater(self.timeout, transport.failIfNotConnected, error.TimeoutError())

self.factory.startedConnecting(self)

进过上面分析,reactor, factory, protocol就联系到一起了,很多操作也就清晰了。再回过去读最前面引用的内容,应该好理解多了。

<二>、对于服务端:reactor. listenTCP(self, port, factory, backlog=50, interface=‘‘)

具体实现如下:

p = tcp.Port(port, factory, backlog, interface, self)

p.startListening()

return p

第一句:实例一个tcp.Port类,该类继承了base.BasePort, _SocketCloser。

实现的方法有:

createInternetSocket(self),startListening(self),_buildAddr(self, (host, port)),

doRead(self),connectionLost(self, reason),getHost(self),doWrite(self)

loseConnection(self, connDone=failure.Failure(main.CONNECTION_DONE))

也有self.factory变量

第二句:做必要的准备,如:create  and  bind  socket,侦听端口,读取数据等。

def startListening(self):

"""Create and bind my socket, and begin listening on it.

This is called on unserialization, and must be called after creating a

server to begin listening on the specified port.

"""

try:

skt = self.createInternetSocket()

skt.bind((self.interface, self.port))

except socket.error, le:

raise CannotListenError, (self.interface, self.port, le)

# Make sure that if we listened on port 0, we update that to

# reflect what the OS actually assigned us.

self._realPortNumber = skt.getsockname()[1]

log.msg("%s starting on %s" % (self.factory.__class__, self._realPortNumber))

# The order of the next 6 lines is kind of bizarre.  If no one

# can explain it, perhaps we should re-arrange them.

self.factory.doStart()

skt.listen(self.backlog)

self.connected = True

self.socket = skt

self.fileno = self.socket.fileno

self.numberAccepts = 100

self.startReading()

其中skt是socket.socket()返回的socket句柄。

三、继续深入:

上面讲到的是reactor.connectTCP(),这个方法会直接开始工作的(初始化factory,protocol等),也许我们需要自己手工控制这个过程。下面是利用类去实现:

\appliction\internet.py文件中的,类TCPClient,当时为了找到这个类名,花了很多时间,这个类名是通过组合而成。

其原型是:class _AbstractClient(_VolatileDataService)

还有class _AbstractServer(_VolatileDataService)作为服务端的原型

一下几个类均是上面两个类的一个模式:

TCPServer, TCPClient,

UNIXServer, UNIXClient,

SSLServer, SSLClient,

UDPServer, UDPClient,

UNIXDatagramServer, UNIXDatagramClient,

MulticastServer

class _AbstractClient和class _AbstractServer最终都是从\appliction\server.py中Server继承而来的。两个类都是服务,只是在实现过程稍有差别,一个针对connect,一个针对port。

它们均有:

self.reactor,是通过参数传递进去的。

Self.method,保存的是连接的方式,如tcp,udp

self._connection,保存连接的(其实是上面的listen**或connect**的返回值)

已经实例了对象,如何开始服务(工作)呢? startService(self)

1)、先来看看class _AbstractServer中的:

def startService(self):

service.Service.startService(self)

if self._port is None:

self._port = self._getPort()

def _getPort(self):

"""

Wrapper around the appropriate listen method of the reactor.

@return: the port object returned by the listen method.

@rtype: an object providing L{IListeningPort}.

"""

if self.reactor is None:

from twisted.internet import reactor

else:

eactor = self.reactor

return getattr(reactor, ‘listen%s‘ % (self.method,))(

*self.args, **self.kwargs)

Getattr()这个函数不错,呵呵

最终调用的还是reactor中的linsten**,但是经过是先定义一个server,再通过server.startServer()开启的。

2)、再来看看class _AbstractClient

def startService(self):

service.Service.startService(self)

self._connection = self._getConnection()

def _getConnection(self):

"""

Wrapper around the appropriate connect method of the reactor.

@return: the port object returned by the connect method.

@rtype: an object providing L{IConnector}.

"""

if self.reactor is None:

from twisted.internet import reactor

else:

reactor = self.reactor

return getattr(reactor, ‘connect%s‘ % (self.method,))(

*self.args, **self.kwargs)

参照上面的,很清晰了。

时间: 2024-08-06 11:56:14

Python中reactor,factory,protocol的相关文章

python中基于descriptor的一些概念(下)

@python中基于descriptor的一些概念(下) 3. Descriptor介绍 3.1 Descriptor代码示例 3.2 定义 3.3 Descriptor Protocol(协议) 3.4 Descriptor调用方法 4. 基于Descriptor实现的功能 4.1 property 4.2 函数和方法,绑定与非绑定 4.3 super 5. 结尾 3. Descriptor介绍 3.1 Descriptor代码示例 class RevealAccess(object):   

python中的特殊数据类型

一.python中的特殊数据类型 对于python,一切事物都是对象,对象基于类创建.像是"wangming",38,[11,12,22]均可以视为对象,并且是根据不同的类生成的对象. 参照:http://www.cnblogs.com/wupeiqi/articles/4911365.html 1.列表 如[12,12,23].['wan','fad','dfjap]等 列表具备的功能: class list(object): """ list() -&

python中os.system()的返回值

最近遇到os.system()执行系统命令的情况,上网搜集了一下资料,整理如下,以备不时之需,同时也希望能帮到某些人. 一.python中的 os.system(cmd)的返回值与linux命令返回值(具体参见本文附加内容)的关系 大家都习惯用os.systemv()函数执行linux命令,该函数的返回值十进制数(分别对应一个16位的二进制数).该函数的返回值与 linux命令返回值两者的转换关系为:该函数的返回值(十进制)转化成16二进制数,截取其高八位(如果低位数是0的情况下,有关操作系统的

Python中pickle模块的使用方法详解

python的pickle模块实现了基本的数据序列和反序列化.通过pickle模块的序列化操作我们能够将程序中运行的对象信息保存到文件中去,永久存储:通过pickle模块的反序列化操作,我们能够从文件中创建上一次程序保存的对象.本文和大家分享的就是python开发中pickle模块的相关使用,一起来看看吧. 基本接口: pickle.dump(obj, file, [,protocol]) 注解:将对象obj保存到文件file中去. protocol为序列化使用的协议版本,0:ASCII协议,所

python 中关于descriptor的一些知识问题

这个问题从早上日常扫segmentfault上问题开始 有个问题是 class C(object): @classmethod def m(): pass m()是类方法,调用代码如下: C.m() 但我想当成属性的方式调用,像这样: C.m 请问该怎么弄呢? 请最好提供个简单的例子, 多谢! 这里我开始误会了他的意思,以为他是想直接使用C().m调用这个方法,如果是这样,直接将装饰器@classmathod改成@property就可以达到效果了. 但是这里他想要达到的效果是C.m 也就是说在不

Python中sqlite学习教程

Python SQLITE数据库是一款非常小巧的嵌入式开源数据库软件,没有独立的维护进程,所有的维护都来自于程序本身.它使用一个文件存储整个数据库,操作十分方便. 以下是重要的 sqlite3 模块程序,可以满足您在 Python 程序中使用 SQLite 数据库的需求.如果您需要了解更多细节,请查看 Python sqlite3 模块的官方文档. 序号 API & 描述 1 sqlite3.connect(database [,timeout ,other optional arguments

python中的generator(coroutine)浅析和应用

背景知识: 在Python中一个function要运行起来,它在python VM中需要三个东西. PyCodeObject,这个保存了函数的代码 PyFunctionObject,这个代表一个虚拟机中的一个函数对象 PyFrameObject,这个代表了函数运行时的调用链和堆栈 Python正是通过这三样东西模拟0x86的函数调用的 在python中 coroutine(协程)被称为的generator,这两个东西在python其实是同一个东东,之所以如此称呼是因为它有迭代器的功能,但是又可以

python中的 descriptor

学好和用好python, descriptor是必须跨越过去的一个点,现在虽然Python书籍花样百出,但是似乎都是在介绍一些Python库而已,对Python语言本身的关注很少,或者即使关注了,但是能够介绍把 dscriptor介绍清楚的,是很少的,到目前,我自己还没有见到过. 一个attr能被称为descriptor,除了需要定义 descriptor protocol 规定的方法外,这个attr必须是属于某个class的,不能是属于某个instance 一.Python中的descript

文件操作,路径操作,StringIO和BytesIO,序列化反序列化,正则表达式与python中使用

文件操作 打开操作open(file, mode='r', buffering=-1, encoding=None, errors=None, newline=None, closefd=True,opener=None)打开一个文件,返回一个文件对象(流对象)和文件描述符.打开文件失败,则返回异常基本使用: 创建一个文件test,然后打开它,用完关闭文件操作中,最常用的操作就是读和写. 文件访问的模式有两种:文本模式和二进制模式.不同模式下,操作函数不尽相同,表现的结果也不一样.open的参数