flask jsonify之序列化时的default函数、jsonify序列化自定义对象[通俗易懂]

flask jsonify之序列化时的default函数、jsonify序列化自定义对象[通俗易懂]目录1.看源码2、重写默认的default函数,实现自己的序列化机制3、把对象转化成字典3.1__dict__的方式3.2、定义keys和__getitem__的方式4、最终的代码实现5、关于default函数的其他知识1.看源码打开site-package,flask,json,__init__.pyjsonify回去调用default()函数,我们…

大家好,又见面了,我是你们的朋友全栈君。

目录

1.看源码

2、重写默认的default函数,实现自己的序列化机制

3、把对象转化成字典

3.1 __dict__的方式

3.2、定义keys和__getitem__的方式

4、最终的代码实现

5、关于default函数的其他知识


1.看源码

打开site-package,flask,json,__init__.py

flask jsonify之序列化时的default函数、jsonify序列化自定义对象[通俗易懂]

jsonify回去调用default()函数,我们最关心的就是重写default方法

我们是不是调用jsonify就一定会调用default呢?答案:不是!

flask jsonify之序列化时的default函数、jsonify序列化自定义对象[通俗易懂]

发现如上图所示,并没有进入jsonify的default方法里,而是直接把字典给序列化出来了。

那么什么时候会调用default呢?

结论:如果flask知道如何序列化你传入进来的数据结构的时候,是不会调用default,因为知道如何序列化就直接帮我们序列化了,但是如果我们要序列化一个对象,是我们的user模型,flask默认是不知道怎么去序列化这个模型的,那么就会去调用default函数,为什么会这样的,原因就在于flask不知道怎么序列化,但是它会给我们一个途径,让我们来指明这个数据结构应该怎么序列化,换句话说,default函数最主要的就是我们需要在内部把不能序列化的结构转化为可以序列化的结构,比如我们传入进来的是一个user,user是不能序列化的,但是如果我们可以把user转化成字典,字典是可以序列化的,那么这样就能完成user对象的序列化了,虽然user作为一个模型他不能序列化,但是我们可以把他的信息读取出来,转化为一个字典,从而保证我们整个序列化的成功执行。可以看到default里面的源码,传入的user对象既不是datetime也不是date、uuid.UUID、__html__,所以最后会抛出一个异常

flask jsonify之序列化时的default函数、jsonify序列化自定义对象[通俗易懂]

flask jsonify之序列化时的default函数、jsonify序列化自定义对象[通俗易懂]

所以我们要在default中把不能序列化的user转化成可以序列化的格式。

所以我们继承,然后重写default方法,在重写的函数中实现user的可序列化就OK了

2、重写默认的default函数,实现自己的序列化机制

我们不要直接修改源码,要在外部继承JSONEncoder,然后在用自己的方法覆盖原来的default方法。

from flask import Flask, jsonify
class hehe:
    name = 'zhangsan'
    age = 18

app = Flask(__name__)
ctx = app.app_context()
ctx.push()
# 上面是解决上下文对象的异常RuntimeError: Working outside of application context.

a = hehe()
print(a)
jsonify(a)  # TypeError: Object of type 'hehe' is not JSON serializable

可以看到上图代码报错不能序列化a对象,所以我们要在外部继承JSONEncoder,然后在用自己的方法覆盖原来的default方法。

flask jsonify之序列化时的default函数、jsonify序列化自定义对象[通俗易懂]

可以看到即使按照上图所示写,flask还是没有调用我们自己定义的default,所以我们还要在flask里面替换一下

from flask import Flask as _Flask, jsonify
from flask.json import JSONEncoder as _JSONEncoder


class JSONEncoder(_JSONEncoder):

    def default(self, o):
        pass


class Flask(_Flask):
    json_encoder = JSONEncoder


class hehe():
    name = 'zhangsan'
    age = 18


app = Flask(__name__)
ctx = app.app_context()
ctx.push()
# 上面是解决上下文对象的异常RuntimeError: Working outside of application context.

a = hehe()
print(a)

jsonify(a)  # 不报错了

flask jsonify之序列化时的default函数、jsonify序列化自定义对象[通俗易懂]

可以看到flask把我们实例化的hehe类当做参数o传递了进来,有两个属性

3、把对象转化成字典

3.1 __dict__的方式

现在我们要把对象转化成字典,因为字典是可以被序列化的, 但是对象不行。我们想到了对象的__dict__内置方法,但是发现没有得到任何的结果,输出的是一个空的json对象。

flask jsonify之序列化时的default函数、jsonify序列化自定义对象[通俗易懂]

这是因为我们在hehe类里面定义的是类的变量而不是实例的变量。类的变量是不会被存放到对象的__dict__当中的。所以加入一个实例变量的时候就有值了(如下图)

flask jsonify之序列化时的default函数、jsonify序列化自定义对象[通俗易懂]

所以我们看到这种方式是可以的,但是我们想把无论类变量还是实例变量都像把它直接序列化,我们就需要用下面的方法把所有的都转成字典

3.2、定义keys和__getitem__的方式

python 对象转字典及序列化对象相关问题,__dict__!!!!必看,多坑

class D:
    name = 'zhangsan'
    age = 18
 
    def __init__(self):
        self.sex = '男'
 
    def keys(self):
        return ('name', 'sex')
 
    def __getitem__(self, item):
        return getattr(self, item)
 
 
d = D()
print(d.__dict__)  # {'sex': '男'}
print(dict(d))  # {'name': 'zhangsan', 'sex': '男'} 如果注销了getitem方法就会报错TypeError: 'D' object is not iterable

4、最终的代码实现

from flask import Flask as _Flask, jsonify
from flask.json import JSONEncoder as _JSONEncoder


class JSONEncoder(_JSONEncoder):
    def default(self, o):
        if hasattr(o, 'keys') and hasattr(o, '__getitem__'):
            print(dict(o))
        else:
            print("不能序列化对象")


class Flask(_Flask):
    json_encoder = JSONEncoder


class hehe:
    name = 'zhangsan'
    age = 18

    def __init__(self):
        self.sex = '男'

    def keys(self):
        return ('name', 'sex')

    def __getitem__(self, item):
        return getattr(self, item)


app = Flask(__name__)
ctx = app.app_context()
ctx.push()
# 上面是解决上下文对象的异常RuntimeError: Working outside of application context.
a = hehe()
print(a)  # <__main__.hehe object at 0x7f0aed3e1e10>
jsonify(a)  # {'name': 'zhangsan', 'sex': '男'}

5、关于default函数的其他知识

default函数是被递归调用的,之所以我们没看到被递归调用是因为我们定义的类都太简单了,如果对象下面的某个属性是另外一个对象的情况

只要遇到不能序列化的对象,都会传入default里面让我们来解决,

例如下面所示default调用了两次:

flask jsonify之序列化时的default函数、jsonify序列化自定义对象[通俗易懂]

flask jsonify之序列化时的default函数、jsonify序列化自定义对象[通俗易懂]

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/142301.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • clion永久激活码[免费获取]

    (clion永久激活码)本文适用于JetBrains家族所有ide,包括IntelliJidea,phpstorm,webstorm,pycharm,datagrip等。IntelliJ2021最新激活注册码,破解教程可免费永久激活,亲测有效,下面是详细链接哦~https://javaforall.net/ide…

    2022年3月22日
    409
  • [软件工程]团队项目–学术家族树之NABC分析

    [软件工程]团队项目–学术家族树之NABC分析

    2021年8月13日
    49
  • 【详细】3分钟教会你使用USDT钱包「建议收藏」

    【详细】3分钟教会你使用USDT钱包「建议收藏」小编为您介绍一款安全系数非常高的USDT钱包,之所以安全系数非常高是因为去中心化模式,USDT钱包文件或私钥可以脱离网络储存,比如可以备份在手机里,硬盘里,U盘里,网盘里,甚至拿笔写在纸上。我们经常听说用户一般会把大部分数字资产存储到冷钱包,主要是为了安全的考虑,而把少量的数字资产存储在交易所热钱包,以方便交易,现在USDT钱包居多的是中心化钱包,用户的数字货币储存在交易所里,换句话说非常不安全,因为你不知道你的USDT钱包文件与私钥,交易所被黑客攻击或者自行关闭你是无能为力只能眼睁睁看这你的币子任由他

    2022年5月21日
    520
  • pycharm汉化版安装[通俗易懂]

    pycharm汉化版安装[通俗易懂]pycharm汉化版安装想要学好一门语言一款好用的编辑软件非常的重要,最近公司要做一款智能机器人的客服聊天系统,用到python刚开始使用eclipse编辑,发现效果不太理想,毕竟不是专业化软件。好了废话少说开启pycharm的安装之旅吧!一、首先呢当然是下载,不过呢我已经准备好了!赶快通过百度云下载吧,链接:百度云链接密码:32g6二、下载完成解压到自己想保存

    2022年5月25日
    48
  • 系统首选dns服务器修改,Windows系统首选DNS如何设置

    系统首选dns服务器修改,Windows系统首选DNS如何设置Windows系统首选DNS如何设置的呢,有时候可能需要修改或者调整DNS服务器地址的设置,以达到优化网络连接速度的效果。该怎么办?下面是学习啦小编收集整理的Windows系统首选DNS如何设置,希望对大家有帮助~~Windows系统首选DNS的设置方法一:在图形界面下设置DNS服务器址1这里Windows8为例,首先在屏幕的右下角找到“网络连接”图标,如图所示2在“网络连接”…

    2022年5月4日
    95
  • insert oracle用法,insert into select的实际用法,insertselect

    insert oracle用法,insert into select的实际用法,insertselectinsertintoselect的实际用法,insertselectINSERTINTOSELECT语句语句形式为:InsertintoTable2(field1,field2,…)selectvalue1,value2,…fromTable1或者:InsertintoTable2select*fromTable1注意:(1)要求目标表Table2必须存…

    2022年7月16日
    15

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号