33

我想避免酸洗类实例中的某些字段。目前,在酸洗之前,我只是将这些字段设置为无,但我想知道是否有更优雅的解决方案?

4

3 回答 3

44

处理不可挑选对象的实例属性的一种方法是使用可用于修改类实例状态的特殊方法:__getstate__()__setstate__(). 这是一个例子

class Foo(object):

    def __init__(self, value, filename):
        self.value = value
        self.logfile = file(filename, 'w')

    def __getstate__(self):
        """Return state values to be pickled."""
        f = self.logfile
        return (self.value, f.name, f.tell())

    def __setstate__(self, state):
        """Restore state from the unpickled state values."""
        self.value, name, position = state
        f = file(name, 'w')
        f.seek(position)
        self.logfile = f

当一个实例被腌制时,Python 将只腌制在调用该实例的方法Foo时返回给它的值。__getstate__()同样,在 unpickling 期间,Python 会将 unpickled 的值作为参数提供给实例的__setstate__()方法。在__setstate__()方法内部,我们能够根据我们腌制的名称和位置信息重新创建文件对象,并将文件对象分配给实例的 logfile 属性。

参考:http ://www.ibm.com/developerworks/library/l-pypers.html

于 2010-02-27T02:14:46.290 回答
19

文档中有一个示例可以解决您的问题__getstate____setstate__

于 2010-02-27T02:00:01.970 回答
17

酸洗使用对象的__getstate____setstate__方法;您可以覆盖它们并忽略您想要的字段。

# foo.py
class Foo:
    def __init__(self):
        self.bar = 1
        self.baz = 2

    def __getstate__(self):
        state = self.__dict__.copy()
        # Don't pickle baz
        del state["baz"]
        return state

    def __setstate__(self, state):
        self.__dict__.update(state)
        # Add baz back since it doesn't exist in the pickle
        self.baz = 0
# main.py
import pickle

from foo import Foo


foo = Foo()
print(f"Foo bar: {foo.bar} baz: {foo.baz}")

new_foo = pickle.loads(pickle.dumps(foo))
print(f"New bar: {new_foo.bar} baz: {new_foo.baz}")

输出:

Foo bar: 1 baz: 2
New bar: 1 baz: 0

您可以在此处找到另一个示例:https ://docs.python.org/3/library/pickle.html#handling-stateful-objects

于 2019-01-11T01:32:26.993 回答