1

如何将此代码从 2.5 移植到 3.4 版本?

from __future__ import with_statement
from ..globals import addonPath
import os, time
import collections

class TailDeque(collections.deque):
    '''Implementation of deque with limited maxlen support for Python 2.5.'''
    def __init__(self, iterable=None, maxlen=20):
        super(TailDeque, self).__init__([])
        self.maxlen = maxlen
        if iterable is not None:
            self.extend(iterable)

    def extend(self, iterable):
        for item in iterable:
            self.append(item)

    def extendleft(self, iterable):
        for item in iterable:
            self.appendleft(item)

    def appendleft(self, item):
        if len(self) == self.maxlen:
            self.pop()
        super(TailDeque, self).appendleft(item)    

    def append(self, item):
        if len(self) == self.maxlen:
            self.popleft()
        super(TailDeque, self).append(item)

logPath = os.path.join( addonPath, "log.txt" )
logQueue = TailDeque()
def log(text):
    return
    logQueue.append(time.strftime("%H:%M:%S") + " - " + text)
    with open(logPath, "w") as fs:
        fs.write('\n'.join(logQueue))

第 10 行:self.maxlen = maxlen

AttributeError:“collections.deque”对象的属性“maxlen”不可写

4

1 回答 1

3

我认为您需要替换这两行:

super(TailDeque, self).__init__([])
self.maxlen = maxlen

和:

if sys.version_info <= (2, 5):
    super(TailDeque, self).__init__([])
    self.maxlen = maxlen
else:
    super(TailDeque, self).__init__([], maxlen=maxlen)

如果你不需要保持与 Python 2.5 的兼容性,你可以简单一点:

super(TailDeque, self).__init__([], maxlen=maxlen)


动机

Python 2.5中,构造函数collections.deque只接受一个参数iterable

deque([iterable])

所以你必须在初始化对象后设置maxlen 。

Python 2.6中,您可以提供maxlen作为可选参数:

collections.deque([iterable[, maxlen]])

这在Python 3.4中保持不变。我认为将其设置为构造函数参数是从 2.6 开始的首选方法。他们无法取消您在 2.x 系列中的方法(直接设置属性),因为他们不想破坏向后兼容性。

在 Python 3 中,没有这样的问题,因此使用构造函数参数成为您可以使用的唯一方法。


边注

与其使用一个空列表进行初始化,然后使用用户提供的 utterable 进行扩展,为什么不collections.deque直接将该迭代传递给呢?IE

super(TailDeque, self).__init__(iterable, maxlen=maxlen)
于 2016-02-11T19:57:48.483 回答