DRF的序列化组件

首先我们要知道序列化是干嘛的,在此之前我们应该知道json格式的数据,一般在前后端交互或者是跨平台交互的时候,会默认使用Json格式拉进行数据的传输,所以当我们把普通的数据转换成json格式的时候就会使用序列化组件,将其序列化成json格式,然后前端接收到json格式之后,再反序列化把json格式数据转换成普通格式的数据,再进行逻辑运算,判断以及渲染页面等.

而DRF的序列化组件,所完成的功能也是如此,负责将对象数据序列化前台所需要的数据,或者反序列化前台的数据,进行校验,来确保数据的安全.

下面我们就介绍三种DRF中最常用的序列化组件,Serializer,ModelSerializer,以及ListModelSerializer.

Serializer组件

在使用Serializer组件之前,我们要先生成一个序列化器,在我们django的项目名(这里我们定义项目名为api)下面新建一个serializer.py文件,在里面继承serializers,生成序列器,供我们在视图函数里面调用:

#/api/models.py
from django.db import models class User(models.Model):
CHOICES_SEX = ((0, '男'), (1, '女'))
name = models.CharField(max_length=64)
pwd = models.CharField(max_length=64, null=True)
age = models.IntegerField(default=0)
height = models.DecimalField(max_digits=5, decimal_places=2, default=0)
icon = models.ImageField(upload_to='icon', default='default.png')
sex = models.IntegerField(choices=CHOICES_SEX, default=0) #/api/serializers.py,在这里生成序列化器
from rest_framework import serializers class UserSerializer(serializers.Serializer):
'''
这里是声明序列化类,都是models.py已有的字段
如果要参与序列化,这里的字段名字一定要和models.py里面的属性同名,如果不参与序列化,就不要在这里声明字段
'''
name = serializers.CharField()
age = serializers.IntegerField()
height = serializers.DecimalField(max_digits=5, decimal_places=2)
'''
下面是自定义序列化字段,序列化的属性值由方法来提供,
方法的名字:固定为 get_属性名,
方法的参数:self为序列化对象,obj为序列化的model对象
注意 : 建议自定义序列化字段名不要与model已有的属性名重名,否则会覆盖model已有字段的声明
注意 : 自定义序列化字段要用SerializerMethodField()作为字段类型
'''
gender = serializers.SerializerMethodField()
def get_gender(self, obj):
return obj.get_sex_display()

在完成以上序列化器生成之后,我们就可以在views.py里面去定义我们的方法,从而来使用序列化器,当然,在views.py定义之前,我们需要在urls.py里面写入路由匹配关系,实例如下,分为序列化和反序列化:

序列化

序列化数据通常是通get请求里面取出来的,常用来查询数据库

# api/urls.py
from django.conf.urls import url
from . import views
urlpatterns = [
url(r'^v1/users/$', views.UserAPIView.as_view()),
url(r'^v1/users/(?P<pk>\d+)/$', views.UserAPIView.as_view()),
] # api/views.py,
'''在视图函数里写我们的业务逻辑,大致分三步:
1. 通过ORM操作数据库取到前端需要的数据
2. 将数据序列化
3. 将序列化后的数据返回给前端
'''
from rest_framework.views import APIView
from rest_framework.response import Response
from django.conf import settings
from . import models class UserAPIView(APIView):
def get(self, request, *args, **kwargs):
pk = kwargs.get('pk')
# 这里是单查的写法,即前端发来的url里面有拼接参数的时候,会走if里面的流程,即我们有匹配条件,通过这些条件去数据库里面查询相应的数值
if pk:
# 1. 通过ORM取数据
user_obj = models.User.objects.filter(pk=pk).first()
if not user_obj:
return Response({
'status': 1,
'msg': '单查 error'
}) # 2. 将数据序列化
# 序列化时里面除了需要序列化的对象以外,还有一个many参数,该参数的意义是: 如果要序列化的数据是单个对象,many=False,如果要序列化的对象数据是多个对象,many=True user_ser = serializers.UserSerializer(user_obj, many=False)
user_data = user_ser.data
# class MySerializer: pass
# user_data = MySerializer(user_obj)
# 3. 把序列化后的数据返回给前端
return Response({
'status': 0,
'msg': '单查 ok',
'results': user_data
}) # 群查,即前端发来的url连接不携带拼接参数,会将所有的数据都查出来
# 1. 通过ORM取数据
user_query = models.User.objects.all()
# 2. 将数据序列化
user_list_data = serializers.UserSerializer(user_query, many=True).data
# 3. 把序列化后的数据返回给前端
return Response({
'status': 0,
'msg': '群查 ok',
'results': user_list_data
})

反序列化

反序列化我们需要生成另一个序列化生成器,反序列化的数据一般是从post请求里面取出来的,且通常用来增加数据库里面的数据

# /api/serializers.py,生成反序列化生成器
class UserDeserializer(serializers.Serializer):
'''
反序列器的生成要注意以下几点:
1. 系统的字段,可以在Field类型中设置系统校验规则,比如(name=serializers.CharField(min_length=3))
2. required校验规则绝对该字段是必校验还是可选校验字段(默认required为True,数据库字段有默认值或可以为空的字段required可以赋值为False)
3. 自定义的反序列字段,设置系统校验规则同系统字段,但是需要在自定义校验规则中(局部、全局钩子)将自定义反序列化字段取出(返回剩余的数据与数据库交互)
4. 局部钩子的方法命名 validate_属性名(self, 属性的value),校验规则为 成功返回属性的value 失败抛出校验错误的异常
5. 全局钩子的方法命名 validate(self, 所有属性attrs),校验规则为 成功返回attrs 失败抛出校验错误的异常
'''
name = serializers.CharField(min_length=3, max_length=64, error_messages={
'required': '姓名必填',
'min_length': '太短',
})
pwd = serializers.CharField(min_length=3, max_length=64) # 系统可选的反序列化字段:没有提供不进行校验(数据库中有默认值或可以为空),提供了就进行校验
age = serializers.IntegerField(min_value=0, max_value=150, required=False)
# 自定义反序列化字段:一定参与校验,且要在校验过程中,将其从入库的数据中取出,剩余与model对应的数据才会入库
re_pwd = serializers.CharField(min_length=3, max_length=64) # 自定义校验规则:局部钩子,全局钩子 # 局部钩子:validate_字段名(self, 字段值)
# 规则:成功返回value,失败抛异常
def validate_aaa(self, value):
if 'g' in value.lower():
raise serializers.ValidationError('名字中不能有g')
return value # 全局钩子:validate(self, 所有校验的数据字典)
# 规则:成功返回attrs,失败抛异常
def validate(self, attrs):
# 取出联合校验的字段们:需要入库的值需要拿到值,不需要入库的需要从校验字段中取出
pwd = attrs.get('pwd')
re_pwd = attrs.pop('re_pwd')
if pwd != re_pwd:
raise serializers.ValidationError({'re_pwd': '两次密码不一致'})
return attrs # create重写,完成入库
def create(self, validated_data):
return models.User.objects.create(**validated_data) # /api/views.py
from rest_framework.views import APIView
from rest_framework.response import Response
from django.conf import settings
from . import models class UserAPIView(APIView):
def post(self, request, *args, **kwargs):
'''
反序列化通常也有三步操作:
1. 从请求对象中拿到前台的数据
2. 校验前台数据是否合法
3. 反序列化成后台Model对象与数据库交互
'''
request_data = request.data
# 反序列化时里面除了需要反序列化的对象以外,同样一个many参数,如果要序列化的数据是单个对象,many=False,如果要序列化的对象数据是多个对象,many=True,不写的话默认是False
user_ser = serializers.UserDeserializer(data=request_data)
# 调用反序列化的校验规则有两种,即系统规则和自定义规则(局部钩子,全局钩子)
result = user_ser.is_valid() if result:
# 校验通过,可以与数据库进行交互:增(create),改(update)
user_obj = user_ser.save()
return Response({
'status': 0,
'msg': 'ok',
'results': serializers.UserSerializer(user_obj).data
})
else:
# 校验失败,返回错误信息
return Response({
'status': 1,
'msg': user_ser.errors
}, status=status.HTTP_400_BAD_REQUEST)

以上就是Serializer组件序列化以及反序列化的用法和注意事项,实际上这是一种较底层的用法,所以我们在生产中常用ModelSerializer来完成数据的序列化和反序列化,整个的代码量会更少,开发效率更高.

ModelSerializer组件

ModelSerializer组件除了Serializer组件所有的功能以外,另外提供了一些功能,比如:

  1. 可以基于模型类自动生成一系列字段,无需手动生成
  2. 可以基于模型类自动为Serializer生成validators,比如uinque_together
  3. ModelSerializer组件会自动实现默认的create()和update()功能,无需手动去实现

序列化和反序列化

# ModelSerializer可以将序列化和反序列化的功能整合成一个类,这个类继承自rest_framework.serializers.ModelSerializer

# api/serializers.py
from rest_framework.serializers import ModelSerializer
from . import models class UserModelSerializer(ModelSerializer):
'''
该生成器包括三个部分:
1. Meta子类:
里面用model来绑定关联model表
用fields来设置所有的序列化反序列化字段
用extra_kwargs来设置系统的校验规则,比如长短,报错信息提示等
2. 局部钩子
3. 全局钩子
'''
'''
该生成器里可以完成的事情:
1. 将序列化类与Model类进行绑定
2. 设置序列化与反序列化所有字段(并划分序列化字段与反序列化字段)
3. 设置反序列化的局部钩子与全局钩子
''' # 自定义反序列化字段,校验规则只能在声明自定义反序列化字段时设置,且一定是write_only
re_pwd = serializers.CharField(min_length=3, max_length=64, write_only=True) class Meta:
model = models.User
fields = ['name', 'age', 'height', 'gender', 'pwd', 're_pwd']
extra_kwargs = {
'name': {
'required': True,
'min_length': 3,
'error_messages': {
'min_length': '太短'
}
},
'age': {
'required': True, # 数据库有默认值或可以为空字段,required默认为False
'min_value': 0
},
'pwd': {
'required': True,
'write_only': True, # 只参与反序列化,这里注意,required不能和read_only一起使用,规则会冲突
},
'gender': {
'read_only': True, # 只参与序列化
},
} def validate_name(self, value):
if 'g' in value.lower():
raise serializers.ValidationError('名字中不能有g')
return value def validate(self, attrs):
pwd = attrs.get('pwd')
re_pwd = attrs.pop('re_pwd')
if pwd != re_pwd:
raise serializers.ValidationError({'re_pwd': '两次密码不一致'})
return attrs # 项目名/urls.py
from django.conf.urls import url
from . import views
urlpatterns = [
url(r'^v2/users/$', views.UserV2APIView.as_view()),
url(r'^v2/users/(?P<pk>\d+)/$', views.UserV2APIView.as_view()),
] #/api/views.py
from rest_framework.views import APIView class UserV2APIView(APIView): def get(self, request, *args, **kwargs):
pk = kwargs.get('pk')
# 单查,即url链接有拼接内容
if pk:
user_obj = models.User.objects.filter(pk=pk).first()
if not user_obj:
return Response({
'status': 1,
'msg': '单查 error'
}) # 完成序列化
user_data = serializers.UserModelSerializer(user_obj, many=False).data return Response({
'status': 0,
'msg': '单查 ok',
'results': user_data
}) # 群查,url链接没有拼接内容,全查
user_query = models.User.objects.all()
# 完成序列化
user_list_data = serializers.UserModelSerializer(user_query, many=True).data
return Response({
'status': 0,
'msg': '群查 ok',
'results': user_list_data
}) # 单增
def post(self, request, *args, **kwargs):
request_data = request.data user_ser = serializers.UserModelSerializer(data=request_data) # 校验失败,直接抛异常,反馈异常信息给前台,只要校验通过,代码才会往下执行
result = user_ser.is_valid()
if result:
user_obj = user_ser.save()
return Response({
'status': 0,
'msg': 'ok',
'results': serializers.UserModelSerializer(user_obj).data
})
else:
# 校验失败,返回错误信息
return Response({
'status': 1,
'msg': user_ser.errors
}, status=status.HTTP_400_BAD_REQUEST)

自定义Response方法

在上面的序列化与反序列化方法中,我们多次用到return Response({})返回值,且里面的内容有诸多相似,经常都有status,msg,results等,所以其实我们可以自定义一个Response方法,继承自原来DRF的response方法,并对其做二次封装,类似于我们之前所了解的把相似类提取出来生成一个基类,别的都继承自该基类即可.

所以我们在应用名下面新建一个response.py文件,用以写我们二次封装的Response方法,如下:

# api/response.py
from rest_framework.response import Response # 这里导入DRF原本的Response,并在下面作为父类导入,继承 class APIResponse(Response):
'''
这里我们继承自父类的init,然后重写其中的__init__,其中status和msg给默认值,然后其余默认值均为None,最后的**kwargs可以接收多余的所有键值对并传给前端
'''
def __init__(self,status=0,msg='ok',results=None,http_status=None,headers=None,exception=False,**kwargs):
# data里面所写的是response的基础数据状态码和数据状态信息
data = {
'status':status,
'msg':msg
}
# results是后端传给前端的数据,如果有的话,就在data里面添加,一起返回给前端,如果没有就不添加
if results is not None:
data['results']=results
# 更新**kwargs里面接收到的所有键值对,并添加到data里面
data.update(**kwargs)
# 下面是直接调用父类的init方法,然后把相应的数据赋值进去
super().__init__(data=data,status=http_status,headers=headers,exception=exception) # 在封装完我们自己的response之后,我们就可以在View.py方法里面导入然后直接使用了
# /api/views.py
from .response import APIResponse
from rest_framework import status return Response({
'status': 1,
'msg': user_ser.errors
}, status=status.HTTP_400_BAD_REQUEST)
# 上面这个就可以改成下面这种一行的形式,可以极大简化代码量.
return APIResponse(1,user_ser.errors,http_status=status.HTTP_400_BAD_REQUEST)

基表相关

基表和基类的概念比较相似,其实就是定义一个继承自models.Model的类,然后用到的表类里面都继承该基表,从而减少代码量,节省操作.

基表配置的关键属性是abstract=True,且要定义在内部的Meta类里面,实例如下:

# api/models.py
class BaseModel(models.Model):
create_time = models.DateTimeField(auto_now_add=True)
class Meta:
# 该属性就表示该表是基表,可以供普通Model类继承使用
# 另外还有一点就是,设置了abstract的表类不会在执行数据库迁移命令(makemigration | migrate)的时候新建表
abstract = True

DRF中ORM的多表关联操作

外键设计

跳出DRF这个概念来说的话,ORM的多表关联我们应该是知道一些的,多表关系一共三种,其外键常存在的位置也不尽相同,如下:

  1. 一对多关系:外键放在多的一方,也就是一对多关系中多的那一方
  2. 多对多关系:外键放在常用的一方
  3. 一对一关系:外键放在不常用的一方

而跨表操作我们通常也有一个口诀,就是正向跨表直接点属性,反向跨表表名小写加属性.

DRF中的跨表操作有些区别,使用起来更加简单,不过models里面定义的时候需要加上related_name反向查询字段,实例如下:

# api/models.py
class Author(BaseModel):
name = models.CharField(max_length=16)
class AuthorDetail(BaseModel):
mobile = models.CharField(max_length=11)
# 这里正向查询点Author,反向查询直接点detail即可,
author = models.OneToOneField(to='Author',related_name='detail')

断级联

级联我们都知道是什么,级联的意义在于两个有级联关系的表,一旦一个数据被删除了,跟其级联相关的另一个表的数据也会被删除,实际上这对于数据的增删来说是非常麻烦的一件事,所以在DRF中我们要执行断级联这种操作,断级联的优点在于:

  1. 表与表之间不会再有外键关联,但是有逻辑关联
  2. 断级联之后不会影响数据库查询表的效率,但是会极大的提高数据库的增删改的效率
  3. 断级联之后一定要通过逻辑代码来保证表与表之间数据的安全,而且有特定的关键字on_delete来表示其不同的级联关系

DRF的models.py里面断级联的参数为db_constraint,我们将其设置为False即可断级联,不同级联关系的表示方式如下:

# /api/models.py
'''
我们假设四种情况,可以用四种on_delete级联方式:
'''
# 1. on_delete=models.CASCADE,作者和作者详情表是一对一级联,且详情表会随着作者表的删除而删除,就需要在详情表里面的author字段里加入该属性
class AuthorDetail(BaseModel):
author=models.OneToOneField(to='Author',db_constraint=False,on_delete=models.CASCADE) # 2. on_delet=models.DO_NOTHING,假设有出版社表和图书表,是一对多的关系,且图书不会随着出版社的删除而随之删除,那么就可以设置on_delete=models.DO_NOTHING
class Book(BaseModel):
publish = models.ForeignKey(to='Publish', related_name='books', db_constraint=False, on_delete=models.DO_NOTHING) # 3. null=True,on_delete=models.SET_NULL,假设有部门表和员工表,是一对多的关系,员工没有部门,可以为空
class Employee(BaseModel):
section = models.ForeignKey(to='Section',related_name='employee',db_constraint=False,null=True,on_delete=models.SET_NULL) # 4. default=0,on_delete=models.SET_DEFAULT,一样有部门表和员工表,在部门表删除的时候,员工不会为空部门,而是自动进入一个默认的部门,即default值
class Employee(BaseModel):
section = models.ForeignKey(to='Section',related_name='employee',db_constraint=False,default=0,on_delete=models.SET_DEFAULT)

注意一点的是,只有一对一和一对多关系的表才有on_delete字段,多对多关系的表没有这个字段,要注意.

DRF的序列化组件的更多相关文章

  1. 第三章、drf框架 - 序列化组件 | Serializer

    目录 第三章.drf框架 - 序列化组件 | Serializer 序列化组件 知识点:Serializer(偏底层).ModelSerializer(重点).ListModelSerializer( ...

  2. Django框架(十八)—— drf:序列化组件(serializer)

    序列化组件 # 模型层 from django.db import models class Book(models.Model): nid = models.AutoField(primary_ke ...

  3. drf框架 - 序列化组件 | Serializer

    序列化组件 知识点:Serializer(偏底层).ModelSerializer(重点).ListModelSerializer(辅助群改) 序列化与反序列化 序列化: 将对象序列化成字符串用户传输 ...

  4. Django框架(十九)—— drf:序列化组件(serializer)

    目录 序列化组件 一.利用for循环来实现序列化(繁琐) 二.利用Django提供的序列化组件(不可控需要的字段) 三.利用drf提供的序列化组件 1.基于Serializer类实现序列化--基本语法 ...

  5. drf 之序列化组件

    序列化 把Python中对象转换为json格式字符串 反序列化 把json格式转为为Python对象. 用orm查回来的数据都是都是一个一个的对象, 但是前端要的是json格式字符串. 序列化两大功能 ...

  6. DRF(3) - 序列化组件(GET/PUT/DELETE接口设计)、视图优化组件

    一.序列化组件 基于上篇随笔的表结构,通过序列化组件的ModelSerializer设计如下三个接口: GET 127.0.0.1:8000/books/{id} # 获取一条数据,返回值:{} PU ...

  7. drf之序列化组件(一):Serializer

    序列化组件:Serializer.ModelSerializer.ListModelSerializer Serializer  偏底层  ModelSerializer       重点  List ...

  8. drf框架 - 序列化组件 | ModelSerializer (查,增,删,改)

    ModelSerializer 序列化准备: 配置 settings.py # 注册rest_framework框架 INSTALLED_APPS = [ ... 'rest_framework' ] ...

  9. DRF(2) - 解析器,序列化组件使用(GET/POST接口设计)

    一.DRF - 解析器 1.解析器的引出 我们知道,浏览器可以向django服务器发送json格式的数据,此时,django不会帮我们进行解析,只是将发送的原数据保存在request.body中,只有 ...

随机推荐

  1. flutter 卡在Running Gradle task 'assembleDebug'...

    Android项目运行时出错 卡在Initializing gradle… 运行时会卡在Initializing gradle..., 此时因为Android项目会用到Gradle, 如果没有FQ,下 ...

  2. CSS template

    ylbtech-CSS3: 1.返回顶部   2.返回顶部   3.返回顶部   4.返回顶部   5.返回顶部     6.返回顶部   作者:ylbtech出处:http://ylbtech.cn ...

  3. checkbox、radio使用jquery改变状态以及其他操作

    $('input[type=checkbox]:checked').each(function(index,elem){ $(elem).attr("checked",false) ...

  4. 从客户端中检测到有潜在危险的 request.form值 以及 request.querystring[解决方法]

    一.从客户端中检测到有潜在危险的request.form值 当页面编辑或运行提交时,出现“从客户端中检测到有潜在危险的request.form值”问题,该怎么办呢?如下图所示: 下面博主汇总出现这种错 ...

  5. Python 操作excel常见异常

    一.使用xlrd模块读取excel: 1.报错:IndexError: list index out of range,如下图 解决方法:reading_sheet.cell(1,0).value中c ...

  6. Jackson环境安装设置

    本地环境设置 由于Jackson是基于Java编程语言的,所以需要设置Java开发环境,这里介绍如何下载安装设置Java.请按照以下步骤来设置环境. Java SE是免费的,点击下载链接:下载Java ...

  7. 天道神诀--IPSAN(iscsi配置)

    数据存储技术 DSA(Direct Attacted Storage 直接附加存储)本地硬盘 NAS(Network Attacted Storage 网络附加存储)网络服务共享:文件夹 SAN(St ...

  8. 异步action和redux-thunk理解

    异步action一般指的就是异步action创建函数 action创建函数分为同步action创建函数和异步action创建函数 同步action创建函数(最常见的): function reques ...

  9. Mariadb 10.2.8版本GTID主从环境搭建以及切换

    1.首先搭建主从 主环境:192.168.1.117 从环境:192.168.1.123 a.首先以二进制包的形式安装好MariaDB (忽略不计) b.配置环境的变量 通配 [mysqld] bin ...

  10. IDEA使用的JDK版本1.9换成1.8后,在IDEA中需要改的配置

    今天上午上课spring5使用注解方式的时候,发现jdk9不兼容,果断换jdk8 步骤如下 一.查看Project中的jdk 1.检查Project SDK:中jdk 版本是否为1.8版本 2.检查P ...