Pickle:AttributeError:'module'对象没有属性

时间:2013-02-27 16:37:19

标签: python attributeerror pickle

我发现了很多关于此问题的线索,但所有这些问题都是命名空间。我的问题与命名空间无关。

一个小例子:

import cPickle as pickle
from uncertainties import Variable

class value(Variable):
    def __init__(self, args, showing=False):
        self.show = showing
        Variable.__init__(self, args[0], args[1])

val = value((3,1), True)
print val.nominal_value, val.std_dev(), val.show
fobj = file("pickle.file", "w")
pickle.dump(val, fobj)
fobj.close()

fobj = file("pickle.file", "r")
val = pickle.load(fobj)
fobj.close()
print val.nominal_value, val.std_dev(), val.show

此代码的输出:

3.0 1.0 True
3.0 1.0
---------------------------------------------------------------------------
AttributeError                            Traceback (most recent call last)
/usr/lib/python2.7/dist-packages/IPython/utils/py3compat.pyc in execfile(fname, *where)
    173             else:
    174                 filename = fname
--> 175             __builtin__.execfile(filename, *where)

/home/markus/pickle.py in <module>()
     19 val = pickle.load(fobj)
     20 fobj.close()
---> 21 print val.nominal_value, val.std_dev(), val.show

AttributeError: 'value' object has no attribute 'show'

在酸洗和去除斑点时,命名空间是相同的。 uncertainties.Variable的所有属性都已恢复 - 只有我添加的一个“节目”被遗漏。

1 个答案:

答案 0 :(得分:6)

uncertainties.Variable() class使用__slots__ attribute来节省内存。因此,要成为pickleable,必须定义__getstate__ method(请参阅Why am I getting an error about my class defining __slots__ when trying to pickle an object?)。

如果您需要添加自己的附加属性,则必须覆盖该__getstate__方法。在您自己的__slot__属性中声明其他属性也是一个好主意:

from uncertainties import Variable

class value(Variable):
    __slots__ = ('show',)  # only list *additional* slots

    def __init__(self, args, showing=False):
        self.show = showing
        super(value, self).__init__(args[0], args[1])

    def __getstate__(self):
        obj_slot_values = {}
        for cls in type(self).mro():
            obj_slot_values.update((k, getattr(self, k)) for k in getattr(cls, '__slots__', ()))
        # Conversion to a usual dictionary:
        return obj_slot_values

这个新的__getstate__是必需的,因为Variable.__getstate__方法假定只有一个 __slots__属性,而MRO中的每个类都可能有一个代替。

这实际上是uncertainties库的限制;我已经提交了pull request来解决此问题,并且无需覆盖子类中的__getstate__方法。