9

我想实现一个符号类型,它跟踪我们已经拥有的符号(保存在 中_sym_table),如果它们存在则返回它们,否则创建新的。编码:

# -*- coding: utf-8 -*-

_sym_table = {}

class Symbol(object):
    def __new__(cls, sym):
        if sym not in _sym_table:
            return super().__new__(cls)
        else:
            return _sym_table[sym]

    def __init__(self, sym):
        self.sym = sym
        _sym_table[sym] = self

    def __str__(self):
        return self.sym

    def __cmp__(self, other):
        return self is other

    def __hash__(self):
        return self.sym.__hash__()

但是,当我调用copy.deepcopy此类Symbol实例的列表时,会引发异常:

a = Symbol('a')
b = Symbol('b')
s = [a, b]
t = copy.deepcopy(s)

错误信息:

Traceback (most recent call last):
  File "xxx.py", line 7, in <module>
    t = copy.deepcopy(s)
  File "/usr/lib/python3.2/copy.py", line 147, in deepcopy
    y = copier(x, memo)
  File "/usr/lib/python3.2/copy.py", line 209, in _deepcopy_list
    y.append(deepcopy(a, memo))
  File "/usr/lib/python3.2/copy.py", line 174, in deepcopy
    y = _reconstruct(x, rv, 1, memo)
  File "/usr/lib/python3.2/copy.py", line 285, in _reconstruct
    y = callable(*args)
  File "/usr/lib/python3.2/copyreg.py", line 88, in __newobj__
    return cls.__new__(cls, *args)
TypeError: __new__() takes exactly 2 arguments (1 given)

所以我的问题是:

  • 如何使用自定义__new__方法对这些对象进行深度复制?
  • 以及关于何时以及如何使用的任何建议copy.deepcopy

非常感谢!

4

3 回答 3

9

一个问题是,deepcopy并且copy无法知道要传递哪些参数__new__,因此它们仅适用于不需要构造函数参数的类。

__init__你可以有参数的原因__init__是在复制对象时不调用它,但__new__必须调用它来创建新对象。

所以如果你想控制复制,你必须定义特殊的__copy____deepcopy__方法:

def __copy__(self):
    return self

def __deepcopy__(self, memo):
    return self

顺便说一句,单例邪恶的,在 python 中并不真正需要。

于 2012-05-16T16:25:24.340 回答
1

在我看来,您希望 Symbol 实例是单例。但是,当您想要一个实例的精确副本(即与原始实例相同的不同实例)时,应该使用 Deepcopy。

所以这里的用法有点与 deepcopy 的目的相矛盾。如果你想让它工作,你可以在Symbol上定义__deepcopy__方法。

于 2012-05-16T15:21:23.813 回答
0

定义__getnewargs__——这样你不仅可以copydeepcopy,而且还可以pickle

于 2015-03-25T08:59:05.097 回答