1

我有一个 Twisted 客户端应用程序,每分钟可以建立数百个连接。我发现我的应用程序存在内存泄漏,我几乎可以肯定这与永远不会删除的 ClientFactory() 派生类有关。

我通过修改 Twisted 文档中的 Echo 客户端示例重现了该问题:

from twisted.internet.protocol import Protocol, ClientFactory
from twisted.internet import reactor
from twisted.internet.task import LoopingCall

from sys import stdout

class Echo(Protocol):
    def connectionMade(self):
        print 'MADE'
        self.transport.write('XXXX')

    def dataReceived(self, data):
        print 'RECV', data
        self.transport.loseConnection()

    def __del__(self):
        print 'DEL PROTOCOL'

class EchoClientFactory(ClientFactory):
    def startedConnecting(self, connector):
        print 'Started to connect.'

    def buildProtocol(self, addr):
        print 'Connected.'
        return Echo()

    def clientConnectionLost(self, connector, reason):
        print 'Lost connection.  Reason:', reason

    def clientConnectionFailed(self, connector, reason):
        print 'Connection failed. Reason:', reason

    def __del__(self):
        print 'DEL FACTORY'

def connector():
    print 'CONNECTOR'
    factory = EchoClientFactory()
    reactor.connectTCP('localhost', 7, factory)

#reactor.callLater(2, connector)
register_loop = LoopingCall(connector)
register_loop.start(1)

reactor.run()

使用此代码,我发现 EchoClientFactory() 的实例仅在程序关闭时才被删除。连接完成时不会删除它们。如果我需要做一些事情来删除工厂实例,我还没有在文档中找到。

4

1 回答 1

1

向协议和工厂添加__del__方法会使它们无法收集。查看gc.garbage列表。在这个例子中,这个列表将永远增长。如果您删除这些__del__方法,问题就会消失。在我的 64 位 Ubuntu 12.04 机器上,该客户端的常驻内存使用量稳定在 13MB。

于 2013-01-22T21:53:35.013 回答