使用simplejson
无法序列化任意对象。你需要向dump
和load
传递一个default
和object_hook
。以下是一个示例:
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
但是,你真正需要的是一个名为default
的函数,它能够从你想要序列化的对象中创建字典,以及一个名为object_hook
的函数,当给定字典时返回一个正确类型的对象(如果仅仅有字典是不够的)。最佳方法是在可序列化类上有一些方法,它们可以从对象创建一个字典,并且可以将其构建回来,同时还应该有一个映射,用于识别字典属于哪个类。
你也可以向类中添加标识符,用作_classes
的索引。这样,如果必须移动一个类,则不会出现问题。