0

我想深度复制一个类的所有实例,但不复制给定类的定义和规则。

预期用途是,假设我有“OriginalClass”类,它的实例是我存储的变量。当程序继续运行时,我希望能够使用 OriginalClass 以前的“快照”,我将只使用其中的值(我不会在那里存储新的值,因此不需要继承任何规则)。

上面的简单示例有效,并且与我想要做的非常相似:

import copy

class VM:
    #nothing in the class
    pass

VM1=VM()
VM1.test=dict()
VM1.test['one']="hello"

def printmytext(VMPast=copy.deepcopy(VM1)):

    g.es(VMPast.test['one'])

VM1.test="Bye"

printmytext() #Returns "hello"

但这个想法是替代:

VMPast=copy.deepcopy(VM1)

对于我不知道如何编写的工作代码,它将复制实例和字典,但不会复制定义。

原因是,在我的程序中使用 deepcopy 版本时(这样:)

VMPast=copy.deepcopy(VM1)
VM1.MyMethod(VMPast.MyButton[-1])

它引发以下错误:

AttributeError: AttributeSetter instance has no __call__ method

因为价值:

VMPast.MyButton[-1]

被发送为:

'__deepcopy__'

而不是我想要的实际替换值......(按钮),所以如果 VMPast 已经是 VM 内实例的工作副本,那么这将在返回deepcopy的情况下起作用!另一件事是,实例是动态的,我事先不知道原始类在那一刻将具有的名称或值。非常感谢!

4

2 回答 2

1

深拷贝从不复制类定义,只复制实例属性。

相反,它会创建一个新实例。Python 实例仅包含对类定义的引用。只需调用MyMethod 实例:

VMPast.MyMethod()

只要类有这样的方法。

阻止成功的深拷贝是您的特定

class VariableManagerClass:
    def __getattr__(self, attr):
        return AttributeSetter(self, attr)

该函数在实例上copy.deepcopy()查找方法,而您的钩子则返回一个类。.__deepcopy__()__getattr__AttributeSetter

您可以通过attr先测试一下来防止这种情况:

class VariableManagerClass:
    def __getattr__(self, attr):
        if attr.startswith('_'):
            raise AttributeError(attr)
        return AttributeSetter(self, attr)

这表明您的类不处理私有属性或特殊方法名称。

于 2013-07-19T11:17:33.863 回答
0

如果您只关心类的值(属性),我认为您可以调用该__dict__方法来获取这些值。

In [1]: class Foo():
   ...:     pass
   ...: 

In [2]: foo = Foo()

In [3]: foo.bar1 = 'derp'

In [4]: foo.bar2 = 'derp derp'

In [5]: foo.__dict__
Out[5]: {'bar1': 'derp', 'bar2': 'derp derp'}

如果这是要问的,这里有一个相关的问题:List attributes of an object

于 2013-07-19T11:26:35.843 回答