在Python的世界里,将一个对象以json格式进行序列化或反序列化一直是一个问题。Python标准库里面提供了json序列化的工具,我们可以简单的用json.dumps来将一个对象序列化。但是这种序列化仅支持python内置的基本类型,对于自定义的类,我们将得到Object of type A is not JSON serializable的错误。

有很多种方法可以用来支持这种序列化,这里有一个很长的关于这个问题的讨论。总结起来,基本上有两种还不错的思路:

  1. 利用标准库的接口:从python标准json库中的JSONDecoder继承,然后自定义实现一个default方法用来自定义序列化过程

  2. 利用第三方库实现:如jsonpickle jsonweb json-tricks

利用标准库的接口的问题在于,我们需要对每一个自定义类都实现一个JSONDecoder.default接口,难以实现代码复用。

利用第三方库,对我们的代码倒是没有任何侵入性,特别是jsonpickle,由于它是基于pickle标准序列化库实现,可以实现像pickle一样序列化任何对象,一行代码都不需要修改。

但是我们观察这类第三方库的输出的时候,会发现所有的这些类库都会在输出的json中增加一个特殊的标明对象类型的属性。这是为什么呢?Python是一门动态类型的语言,我们无法在对象还没有开始构建的时候知道对象的某一属性的类型信息,为了对反序列化提供支持,看起来确实是不得不这么做。

有人可能觉得这也无可厚非,似乎不影响使用。但是在跨语言通信的时候,这就成为了一个比较麻烦的问题。比如我们有一个Python实现的API,客户端发送了一个json请求过来,我们想在统一的一个地方将json反序列化为我们Python代码的对象。由于客户端不知道服务器端的类型信息,json请求里面就没法加入这样的类型信息,这也就导致这样的类库在反序列化的时候遇到问题。

能不能有一个相对完美的实现呢?先看一下我们理想的json序列化库的需求:

  1. 我们希望能简单的序列化任意自定义对象,只添加一行代码,或者不加入任何代码

  2. 我们希望序列化的结果不加入任何非预期的属性

  3. 我们希望能按照指定的类型进行反序列化,能自动处理嵌套的自定义类,只需要自定义类提供非常简单的支持,或者不需要提供任何支持

  4. 我们希望反序列化的时候能很好的处理属性不存在的情况,以便在我们加入某一属性的时候,可以设置默认值,使得旧版本的序列化结果可以正确的反序列化出来

如果有一个json库能支持上面的四点,那就基本是比较好用的库了。下面我们来尝试实现一下这个类库。

对于我们想要实现的几个需求,我们可以建立下面这样的测试来表达我们所期望的库的API设计:

  1. class SerializableModelTest(unittest.TestCase):
  2. def test_model_serializable(self):
  3. class A(SerializableModel):
  4. def __init__(self, a, b):
  5. super().__init__()
  6. self.a = a
  7. self.b = b if b is not None else B(0)
  8. @property
  9. def id(self):
  10. return self.a
  11. def _deserialize_prop(self, name, deserialized):
  12. if name == 'b':
  13. self.b = B.deserialize(deserialized)
  14. return
  15. super()._deserialize_prop(name, deserialized)
  16. class B(SerializableModel):
  17. def __init__(self, b):
  18. super().__init__()
  19. self.b = b
  20. self.assertEqual(json.dumps({'a': 1, 'b': {'b': 2}, 'long_attr': None}), A(1, B(2)).serialize())
  21. self.assertEqual(json.dumps({'a': 1, 'b': None}), A(1, None).serialize())
  22. self.assertEqual(A(1, B(2)), A.deserialize(json.dumps({'a': 1, 'b': {'b': 2}})))
  23. self.assertEqual(A(1, None), A.deserialize(json.dumps({'a': 1, 'b': None})))
  24. self.assertEqual(A(1, B(0)), A.deserialize(json.dumps({'a': 1})))

这里我们希望通过继承的方式来添加支持,这将在反序列化的时候提供一个好处。因为有了它我们就可以直接使用A.deserialize方法来反序列化,而不需要提供任何其他的反序列化函数参数,比如这样json.deserialize(serialized_str, A)

同时为了验证我们的框架不会将@property属性序列化或者反序列化,我们特意在类A中添加了这样一个属性。

由于在反序列化的时候,框架是无法知道某一个对象属性的类型信息,比如测试中的A.b,为了能正确的反序列化,我们需要提供一点简单的支持,这里我们在类A中覆盖实现了一个父类的方法_deserialize_prop对属性b的反序列化提供支持。

当我们要反序列化一个之前版本的序列化结果时,我们希望能正确的反序列化并使用我们提供的默认值作为最终的反序列化值。

如果能有一个类可以让上面的测试通过,相信那个类就是我们所需要的类了。这样的类可以实现为如下:

  1. class ModelBase:
  2. @staticmethod
  3. def is_normal_prop(obj, key):
  4. is_prop = isinstance(getattr(type(obj), key, None), property)
  5. is_constant = re.match('^[A-Z_0-9]+$', key)
  6. return not (key.startswith('__') or callable(getattr(obj, key)) or is_prop or is_constant)
  7. @staticmethod
  8. def is_basic_type(value):
  9. return value is None or type(value) in [int, float, str, list, tuple, bool, dict]
  10. def _serialize_prop(self, name):
  11. value = getattr(self, name)
  12. if isinstance(value, (tuple, list)):
  13. try:
  14. json.dumps(value)
  15. return value
  16. except Exception:
  17. return [v._as_dict() for v in value]
  18. return value
  19. def _as_dict(self):
  20. keys = dir(self)
  21. props = {}
  22. for key in keys:
  23. if not ModelBase.is_normal_prop(self, key):
  24. continue
  25. value = self._serialize_prop(key)
  26. if not (ModelBase.is_basic_type(value) or isinstance(value, ModelBase)):
  27. raise Exception('unkown value to serialize to dict: key={}, value={}'.format(key, value))
  28. props[key] = value if self.is_basic_type(value) else value._as_dict()
  29. return props
  30. def _short_prop(self, name):
  31. value = getattr(self, name)
  32. if isinstance(value, (tuple, list)):
  33. try:
  34. json.dumps(value)
  35. return value
  36. except Exception:
  37. return [v._as_short_dict() for v in value]
  38. return value
  39. def _as_short_dict(self):
  40. keys = dir(self)
  41. props = {}
  42. for key in keys:
  43. if not ModelBase.is_normal_prop(self, key):
  44. continue
  45. value = self._short_prop(key)
  46. if not (ModelBase.is_basic_type(value) or isinstance(value, ModelBase)):
  47. raise Exception('unkown value to serialize to short dict: key={}, value={}'.format(key, value))
  48. props[key] = value if self.is_basic_type(value) else value._as_short_dict()
  49. return props
  50. def serialize(self):
  51. return json.dumps(self._as_dict(), ensure_ascii=False)
  52. def _deserialize_prop(self, name, deserialized):
  53. setattr(self, name, deserialized)
  54. @classmethod
  55. def deserialize(cls, json_encoded):
  56. if json_encoded is None:
  57. return None
  58. import inspect
  59. args = inspect.getfullargspec(cls)
  60. args_without_self = args.args[1:]
  61. obj = cls(*([None] * len(args_without_self)))
  62. data = json.loads(json_encoded, encoding='utf8') if type(json_encoded) is str else json_encoded
  63. keys = dir(obj)
  64. for key in keys:
  65. if not ModelBase.is_normal_prop(obj, key):
  66. continue
  67. if key in data:
  68. obj._deserialize_prop(key, data[key])
  69. return obj
  70. def __str__(self):
  71. return self.serialize()
  72. def _prop_eq(self, name, value, value_other):
  73. return value == value_other
  74. def __eq__(self, other):
  75. if other is None or other.__class__ is not self.__class__:
  76. return False
  77. keys = dir(self)
  78. for key in keys:
  79. if not ModelBase.is_normal_prop(self, key):
  80. continue
  81. value, value_other = getattr(self, key), getattr(other, key)
  82. if not (ModelBase.is_basic_type(value) or isinstance(value, ModelBase)):
  83. raise Exception('unsupported value to compare: key={}, value={}'.format(key, value))
  84. if value is None and value_other is None:
  85. continue
  86. if (value is None and value_other is not None) or (value is not None and value_other is None):
  87. return False
  88. if not self._prop_eq(key, value, value_other):
  89. return False
  90. return True
  91. def short_repr(self):
  92. return json.dumps(self._as_short_dict(), ensure_ascii=False)

为了更进一步提供支持,我们将最终的类命名为ModelBase,因为通常我们要序列化或反序列化的对象都是我们需要特殊对待的对象,且我们通常称其为模型,我们一般也会将其放在一个单独models模块中。

作为一个模型的基类,我们还添加了一些常用的特性,比如:

  1. 支持标准的格式化接口__str__,这样我们在使用'{}'.format(a)的时候,就可以得到一个更易于理解的输出

  2. 提供了一个缩短的序列化方式,在我们有时候不想直接输出某一个特别长的属性的时候很有用

  3. 提供了基于属性值的比较方法

  4. 自定义类的属性可以为基础的Python类型,或者由基础Python类型构成的list tuple dict

在使用这个类的时候,当然也是有一些限制的,主要的限制如下:

  1. 当某一属性为自定义类的类型的时候,需要子类覆盖实现_deserialize_prop方法为反序列化过程提供支持

  2. 当某一属性为由自定义类构成的一个list tuple dict复杂对象时,需要子类覆盖实现_deserialize_prop方法为反序列化过程提供支持

  3. 简单属性必须为python内置的基础类型,比如如果某一属性的类型为numpy.float64,序列化反序列化将不能正常工作

虽然有上述限制,但是这正好要求我们在做模型设计的时候保持克制,不要将某一个对象设计得过于复杂。比如如果有属性为dict类型,我们可以将这个dict抽象为另一个自定义类型,然后用类型嵌套的方式来实现。

到这里这个基类就差不多可以支撑我们日常的开发需要了。当然对于这个简单的实现还有可能有其他的需求或者问题,大家如有发现,欢迎留言交流。

来源:华为云社区原创 作者:Bright Liao

#华为云·寻找黑马程序员# 如何实现一个优雅的Python的Json序列化库的更多相关文章

  1. #华为云·寻找黑马程序员#【代码重构之路】如何“消除”if/else

    1. 背景 if/else是高级编程语言中最基础的功能,虽然 if/else 是必须的,但滥用 if/else,特别是各种大量的if/else嵌套,会对代码的可读性.可维护性造成很大伤害,对于阅读代码 ...

  2. 大型情感剧集Selenium:1_介绍 #华为云·寻找黑马程序员#

    学习selenium能做什么? 很多书籍.文章中是这么定义selenium的: Selenium 是开源的自动化测试工具,它主要是用于Web 应用程序的自动化测试,不只局限于此,同时支持所有基于web ...

  3. python让你再也不为文章配图与素材发愁,让高清图片占满你的硬盘! #华为云·寻找黑马程序员#

    欢迎添加华为云小助手微信(微信号:HWCloud002 或 HWCloud003),输入关键字"加群",加入华为云线上技术讨论群:输入关键字"最新活动",获取华 ...

  4. 使用Python开发小说下载器,不再为下载小说而发愁 #华为云·寻找黑马程序员#

    需求分析 免费的小说网比较多,我看的比较多的是笔趣阁.这个网站基本收费的章节刚更新,它就能同步更新,简直不要太叼.既然要批量下载小说,肯定要分析这个网站了- 在搜索栏输入地址后,发送post请求获取数 ...

  5. 爬虫新宠requests_html 带你甄别2019虚假大学 #华为云·寻找黑马程序员#

    python模块学习建议 学习python模块,给大家个我自己不专业的建议: 养成习惯,遇到一个模块,先去github上看看开发者们关于它的说明,而不是直接百度看别人写了什么东西.也许后者可以让你很快 ...

  6. #华为云·寻找黑马程序员#微服务-你真的懂 Yaml 吗?

    在Java 的世界里,配置的事情都交给了 Properties,要追溯起来这个模块还是从古老的JDK1.0 就开始了的. "天哪,这可是20年前的东西了,我居然还在用 Properties. ...

  7. #华为云·寻找黑马程序员#【代码重构之路】使用Pattern的正确姿势

    1.问题 在浏览项目时,发现一段使用正则表达式的代码 这段代码,在循环里执行了Pattern.matches()方法进行正则匹配判断. 查看matches方法的源码,可以看到 每调用一次matches ...

  8. 三伏天里小试牛刀andriod 开发 #华为云·寻找黑马程序员#

    2019年07月,北京,三伏天,好热啊.越热自己还越懒得动换(肉身给的信号),但是做为产品经理/交互设计师的,总想着思考些什么(灵魂上给的信号),或者是学习些什么,更有利于将来的职业发展吧,哈哈哈.工 ...

  9. 使用jieba分析小说太古神王中,男主更爱谁?去文章中找答案吧!#华为云·寻找黑马程序员#

    欢迎添加华为云小助手微信(微信号:HWCloud002 或 HWCloud003),输入关键字"加群",加入华为云线上技术讨论群:输入关键字"最新活动",获取华 ...

随机推荐

  1. NOIP模拟 29

    T1第一眼觉得是网络流 看见4e6条边200次增广我犹豫了 O(n)都过不去的赶脚.. 可是除了网络流板子我还会什么呢 于是交了个智障的EK 还是用dijkstra跑的 居然有50分!$(RP--)$ ...

  2. Mokia(三维偏序)P4390

    提到cdq,就不得不提这道该死的,挨千刀的题目了. 极简题面: 给定一个二维平面,在ti时刻会在(xi,yi)放一个点,会在tj时刻查询一个方框里面的点的数量 看道题就是二维线段树乱搞啊,这么水??? ...

  3. Linux 学习(1) | 学习方向导图

    方向导图 文件系统导图  内核导图

  4. solr 本地搭建

      1. 运行 D:\solr-4.7.2\example --> java -jar start.jar   2. 添加插件IK D:\solr-4.7.2\example\solr-weba ...

  5. Pandas进阶笔记 (一) Groupby 重难点总结

    如果Pandas只是能把一些数据变成 dataframe 这样优美的格式,那么Pandas绝不会成为叱咤风云的数据分析中心组件.因为在数据分析过程中,描述数据是通过一些列的统计指标实现的,分析结果也需 ...

  6. 小白学 Python(24):Excel 基础操作(下)

    人生苦短,我选Python 前文传送门 小白学 Python(1):开篇 小白学 Python(2):基础数据类型(上) 小白学 Python(3):基础数据类型(下) 小白学 Python(4):变 ...

  7. Linux 下的这些高效指令,是你快速学习的神器

    Linux是一套免费使用和自由传播的类Unix操作系统,是一个基于POSIX和UNIX的多用户.多任务.支持多线程和多CPU的操作系统.它能运行主要的UNIX工具软件.应用程序和网络协议.它支持32位 ...

  8. 在 ASP.NET Core 项目中使用 MediatR 实现中介者模式

    一.前言  最近有在看 DDD 的相关资料以及微软的 eShopOnContainers 这个项目中基于 DDD 的架构设计,在 Ordering 这个示例服务中,可以看到各层之间的代码调用与我们之前 ...

  9. 忘记Linux登录密码的破解方法

    注意:1.破解方式只限于7.0以后的Linux系统. 2.要注意自己linux系统中有没有开启selinux,如果开启则在后面要建一个名为:autorelabel的隐藏文件.     1.启动Linu ...

  10. WordPress 去掉功能中的 wordpress.org

    WordPress 去掉功能中的 wordpress.org