Python优雅地dumps非标准类型

專 欄


正小歪,Python 工程师,主要负责 Web 开发和日志数据处理。博客文章《真正的 Tornado 异步非阻塞》、《 使用 JWT 让你的 RESTful API 更安全》等多次入选知名技术社区每日精选。《 使用 Shipyard 搭建 Docker 集群》被选入 Dockerone 周报。
个人博客: https://www.hexiangyu.me
GitHub: https://github.com/zhengxiaowai

在用Python编程时很经常做的一件事就是 Python 数据类型和 JSON 数据类型的转换。

但是存在一个明显的问题,JSON 作为一种数据交换格式有固定的数据类型,但是 Python 作为编程语言除了内置的数据类型以外还能编写自定义的数据类型。

墙裂推荐:去看看 JSON 官网对 JSON 的介绍:http://www.json.org/json-zh.html

比如你肯定遇到过类似的问题:

那么问题就来了,如何把各种各样的 Python 数据类型转化成 JSON 数据类型。 
一种很不 pythonic 的做法就是,先转换成某种能和 JSON 数据类型直接转换的值,然后在 dump,这么做很直接很暴力,但是在各种花式数据类型面前就很无力。

Google 是解决问题的重要方式之一,当你一顿搜索过后,你就会发现其实可以在 dumps 时 encode 这个阶段对数据进行转化。

所以你肯定是那么做的,完美地解决了问题。


JSON 的 Encode 过程

文中代码摘自 https://github.com/python/cpython

删除了几乎所有的 docstring,由于代码太长,直接截取了重要片段。可以在片段最上方的链接查看完整的代码。

熟悉 json 这个库的都知道基本只有4个常用的 API,分别是 dump、dumps 和 load、loads。

源码位于 cpython/Lib/json 中

直接看到最后的 return。可以发现如果不提供 cls 默认就使用 JSONEncoder,然后调用该类的实例方法 encode。

encode 方法也十分简单:

可以看出最后的我们得到 JSON 都是 chunks 拼接得到的,chunks 是调用 self.iterencode 方法得到的。

iterencode 方法比较长,我们只关心最后几行。

返回值 _iterencode,是函数中 c_make_encoder 或者 _make_iterencode这两个高阶函数的返回值。

c_make_encoder 是来自 _json 这个 module ,这个 module 是一个 c 模块,我们不去关心这个模块怎么实现的。 
转去研究同等作用的 _make_iterencode 方法。

同样需要关心的只有返回的这个函数,代码里各种 if-elif-else 逐一把内置类型转换成 JSON 类型。 
在对面无法识别的类型时候就使用了 _default() 这个方法,然后递归调用解析各个值。

_default 就是最前面那个被覆盖的 default

到这里就可以完全了解 Python 是如何 encode 成 JSON 数据。

总结一下流程,json.dumps() 调用 JSONEncoder 的实例方法 encode(),随后使用 iterencode() 递归转化各种类型,最后把 chunks 拼接成字符串后返回。

优雅的解决方案

通过前面的流程分析之后,知道为什么继承 JSONEncoder 然后覆盖 default 方法就可以完成自定义类型解析了。

也许你以后需要解析 datetime 类型数据,你可定会那么做:

最后调用父类是 default() 方法纯粹是为了触发异常。

Python 可以使用 singledispatch 来解决这种单泛型问题。

这种写法比较符合设计模式的规范。假如以后有了新的类型,不用再修改ExtendJSONEncoder 类,只需要添加适当的 singledispatch 方法就可以了, 比较 pythonic 。

如果你执意的想在类中添加 singledispatch 可以参考:https://stackoverflow.com/a/24602374/5227020 ,当然我仍然觉得还是不要写在类中比较好。


长按扫描关注Python中文社区,

获取更多技术干货!

    

Python 中 文 社 区

Python中文开发者的精神家园

合作、投稿请联系微信:

pythonpost

— 人生苦短,我用Python —
1MEwnaxmMz7BPTYzBdj751DPyHWikNoeFS


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 这是一个关于Python中json.dumps()函数的问题,它用于将Python对象转换为JSON格式的字符串。它接受多个参数,包括要序列化的Python对象,为Python对象编码时使用的编码方式,以及对于JSON字符串的格式化。 ### 回答2: json.dumps() 是 Python 的一个函数,用于将 Python 对象转换成 JSON 字符串。 JSON 是一种轻量级的数据交换格式,它使得数据在不同的系统之间能够方便地进行传输和共享。Python 中的 json.dumps() 函数可以将 Python 中的数据类型,如字典、列表、元组等,转换为 JSON 字符串形式。这种转换使得数据能够在不同的系统中进行传输,并且在接收方能够正确地解析和使用数据。 json.dumps() 函数的语法如下: json.dumps(obj, *, skipkeys=False, ensure_ascii=True, check_circular=True, allow_nan=True, cls=None, indent=None, separators=None, default=None, sort_keys=False, **kw) 其中,obj 是待转换的 Python 对象,其他参数是用于控制转换行为的可选参数。 * skipkeys:如果设置为 True,则会跳过类型不支持的字典的键,而不是引发异常; * ensure_ascii:如果设置为 False,则可以在输出 JSON 字符串时包含 ASCII 字符; * check_circular:如果设置为 False,则可以允许序列中包含循环引用的对象; * allow_nan:如果设置为 False,则会引发 ValueError,如果序列化的对象中包含 NaN 或 Infinity; * cls:指定用于编码基本数据类型的编码器类,默认为 None; * indent:定义输出格式中缩进的字符串(通常是空格字符串); * separators:指定结果中分隔符的字符串; * default:指定一个自定义的对象转换函数; * sort_keys:如果设置为 True,则按键进行排序。 通过使用 json.dumps() 函数,我们可以将 Python 对象转换为 JSON 字符串,方便在不同系统中进行数据传输和共享。 ### 回答3: json.dumpsPython的一个库函数,用来将Python对象转化为JSON格式的字符串。 在使用json.dumps函数时,我们可以传入一个Python对象作为参数,并且可以通过一些可选参数来进行一些配置。它会将这个Python对象序列化为JSON格式的字符串。 当我们传入一个Python对象时,json.dumps会自动将其转化为对应的JSON格式。例如,如果我们传入一个Python的字典对象,那么json.dumps会将它转化为JSON格式的字典。 此外,我们还可以使用一些可选参数来配置dumps函数的行为。其中最常用的参数是indent,它指定了json.dumps输出格式化后的字符串时的缩进字符数。通过指定indent参数,我们可以更加清晰地看到JSON格式的结构。 另一个重要的可选参数是ensure_ascii,它用于控制是否将ASCII字符转义为Unicode转义序列。如果设置为False,则不会转义ASCII字符,否则会将其转义。 此外,json.dumps还支持其他一些参数和配置,比如排序、跳过空白项等。 总的来说,json.dumps函数是Python中用于将Python对象转化为JSON格式的字符串的函数。它能够方便地将Python对象序列化为JSON格式,从而方便地在不同平台和不同语言之间进行数据交换。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值