4

我在很多需要它的类中使用这种复制方法已经有一段时间了。

class population (list):
def __init__ (self):
    pass

def copy(self):
    return copy.deepcopy(self)

它突然开始产生这个错误:

     File "C:\Python26\lib\copy.py", line 338, in _reconstruct
    state = deepcopy(state, memo)
  File "C:\Python26\lib\copy.py", line 162, in deepcopy
    y = copier(x, memo)
  File "C:\Python26\lib\copy.py", line 255, in _deepcopy_dict
    y[deepcopy(key, memo)] = deepcopy(value, memo)
  File "C:\Python26\lib\copy.py", line 189, in deepcopy
    y = _reconstruct(x, rv, 1, memo)
  File "C:\Python26\lib\copy.py", line 323, in _reconstruct
    y = callable(*args)
  File "C:\Python26\lib\copy_reg.py", line 93, in __newobj__
    return cls.__new__(cls, *args)
TypeError: object.__new__(generator) is not safe, use generator.__new__()
>>> 

包含对第 338、162、255、189 行的引用的行在我在这里复制的“第 338 行”之前重复了很多次。

4

2 回答 2

9

你在克隆生成器吗?无法克隆生成器。

在此处复制 Gabriel Genellina 的答案:


没有办法“克隆”生成器:

py> g = (i for i in [1,2,3])
py> type(g)()
Traceback (most recent call last):
   File "<stdin>", line 1, in <module>
TypeError: cannot create 'generator' instances
py> g.gi_code = code
Traceback (most recent call last):
   File "<stdin>", line 1, in <module>
TypeError: readonly attribute
py> import copy
py> copy.copy(g)
Traceback (most recent call last):
...
TypeError: object.__new__(generator) is not safe, use generator.__new__()
py> type(g).__new__
<built-in method __new__ of type object at 0x1E1CA560>

您可以使用生成器函数来做到这一点,因为它充当“生成器
工厂”,在调用时构建一个新生成器。即使使用 Python C
API,要创建一个生成器,也需要一个框架对象——而且没有办法
“即时”创建一个我知道的框架对象:(

py> import ctypes
py> PyGen_New = ctypes.pythonapi.PyGen_New
py> PyGen_New.argtypes = [ctypes.py_object]
py> PyGen_New.restype = ctypes.py_object
py> g = (i for i in [1,2,3])
py> g2 = PyGen_New(g.gi_frame)
py> g2.gi_code is g.gi_code
True
py> g2.gi_frame is g.gi_frame
True
py> g.next()
1
py> g2.next()
2

g 和 g2 共享相同的执行框架,因此它们不是独立的。在
Python 中创建新框架没有简单的方法:

py> type(g.gi_frame)()
Traceback (most recent call last):
   File "<stdin>", line 1, in <module>
TypeError: cannot create 'frame' instances

可以尝试使用 PyFrame_New - 但这对我来说太神奇了......

于 2009-09-09T19:38:33.987 回答
0

当人们不小心尝试将迭代器克隆到一个类时,这在许多情况下都会发生。例如,在 PIL 中,尝试克隆PixelAccessanImage将引发此错误。

举个例子pixels = image.load()pixels_copy = copy.copy(pixels)您必须复制基础对象,然后生成一个迭代器,而不是尝试做类似的事情。因此,将那段代码替换为pixels_copy = image.copy().load().

于 2017-06-30T07:07:50.600 回答