7

我有一个这样定义的类

class A:
    def __init__(self):
        self.item1 = None
    def __repr__(self):
        return str(self.__dict__)

当我做:

>>> import simplejson
>>> myA = A()
>>> simplejson.dumps(myA)
TypeError: {'item1': None} is not JSON serializable

我找不到原因。

我是否需要向 A 添加任何特定方法以便 simplejson 序列化我的类对象?

4

2 回答 2

10

您不能使用simplejson. 您需要将 a defaultand传递object_hookdumpand load。这是一个例子:

class SerializerRegistry(object):
    def __init__(self):
        self._classes = {}
    def add(self, cls):
        self._classes[cls.__module__, cls.__name__] = cls
        return cls
    def object_hook(self, dct):
        module, cls_name = dct.pop('__type__', (None, None))
        if cls_name is not None:
            return self._classes[module, cls_name].from_dict(dct)
        else:
            return dct
    def default(self, obj):
        dct = obj.to_dict()
        dct['__type__'] = [type(obj).__module__,
                           type(obj).__name__]
        return dct

registry = SerializerRegistry()

@registry.add
class A(object):
    def __init__(self, item1):
        self.item1 = item1
    def __repr__(self):
        return str(self.__dict__)
    def to_dict(self):
        return dict(item1=self.item1)
    @classmethod
    def from_dict(cls, dct):
        return cls(**dct)

s = json.dumps(A(1), default=registry.default)
a = json.loads(s, object_hook=registry.object_hook)

这导致:

>>> s
'{"item1": 1, "__type__": ["__main__", "A"]}'
>>> a
{'item1': 1}

但是您真正需要的是一个default从您希望序列化的对象创建字典的函数,以及一个object_hook在字典不够时返回对象(正确类型)的函数。最好的方法是在可序列化的类上使用方法,从对象创建一个字典并将其构造回来,并且还有一个映射来识别字典属于哪个类。

您还可以将标识符添加到要用作索引的类中_classes。这样,如果您必须移动班级,您就不会遇到问题。

于 2011-01-27T21:29:24.857 回答
3

根据json 模块文档(simplejson 在 Python 2.6 中被采用为 json),您需要扩展json.JSONEncoder该类,覆盖其默认方法以将您的对象转换为可以序列化的类型。似乎没有在您的对象上寻找的方法。

于 2011-01-27T21:20:53.790 回答