在python中提供了标准库json将基本类型的数据转化成json格式,可是在涉及到自定义类型时须要扩展或者复写Encoder(Decoder)来实现,默认状况下json会抛出"TypeErro: xxx is not json serializable"的错误。本文参考了文章Json概述以及python对json的相关操做。
html
对于基本类型的数据直接使用json.dumps和json.loads方法进行序列化和反序列化。 python
import json array = ['d', 'b', 'c', 'a', {'b':100, 'a':'letter a'}] encodestr = json.dumps(array) org_obj = json.loads(encodestr)
JSON | Python |
Object | dict |
Array | list |
String | str |
number(int) | int |
number(real) | float |
true | True |
false | False |
null | None |
json.dumps({'4': 5, '6': 7}, sort_keys=True, indent=4, separators=(',', ': '))
json.dumps函数接受参数default用于指定一个函数,该函数可以把自定义类型的对象转换成可序列化的基本类型。json.loads函数接受参数objecthook用于指定函数,该函数负责吧反序列化后的基本类型对象转换成自定义类型的对象。 json
boy1 = boy('Will', 20) #default method for decode def boydefault(obj): if isinstance(obj, boy): return {'name': obj.name, 'age': obj.age} return obj; def boyhook(dic): print('test') if dic['name']: return boy(dic['name'], dic['age']) return dic boy_encode_str = json.dumps(boy1, default=boydefault) new_boy = json.loads(boy_encode_str, object_hook=boyhook) print(boy_encode_str) print(new_boy)
除此之外json库提供了JSONEncoder和JSONDecoder两个类用于json的序列化和反序列化,能够经过子类实现自定义类的json操做(估计json库内部逻辑也是用的这两个类)。
json.JSONEncoder的主要方法: ide
class BoyEncoder(json.JSONEncoder): def default(self, o): if isinstance(o, boy): return {'name': o.name, 'age': o.age} return json.JSONEncoder.default(o); #override decode method class BoyDecoder(json.JSONDecoder): def decode(self, s): dic = super().decode(s); return boy(dic['name'], dic['age']); #override __init__ method class BoyDecoder2(json.JSONDecoder): def __init__(self): json.JSONDecoder.__init__(self) self.object_hook = boyhook boy_encode_str = json.dumps(boy1, cls=BoyEncoder) new_boy = json.loads(boy_encode_str, cls=BoyDecoder) new_boy2 = json.loads(boy_encode_str, cls=BoyDecoder2) print(boy_encode_str) print(new_boy) print(new_boy2)使用两种中的任意一种均可以达到这个目的,不过请注意上述代码已测试可是省略了一些简单的非必要代码,若是有朋友想测试请自行补充:)。