在序列化类中,如果想使用request,则可以通过self.context['request']获取

序列化器的主要由两大功能

- 对请求的数据进行校验(底层调用的是Django的Form和ModelForm)
- 对数据库查询的数据进行序列化

1.数据的校验

注意自定义的钩子函数中的参数value,对于FK或者M2M的字段,是一个对象而不是文本

1.1Serializer

基于Serializer类,类似Django中的Form

import re
from rest_framework.views import APIView
from rest_framework.views import Response
from rest_framework import serializers
from django.core.validators import EmailValidator class MyEmailValidator(object):
def __init__(self, base):
self.base = base def __call__(self, value):
match_object = re.match(self.base, value)
if not match_object:
raise serializers.ValidationError('格式错误') class UserSerializer(serializers.Serializer):
username = serializers.CharField(label='姓名', min_length=6, max_length=32)
age = serializers.IntegerField(label='年龄', min_value=0, max_value=200)
level = serializers.ChoiceField(choices=((1, '董事长'), (2, '经理'), (3, '员工')))
email1 = serializers.EmailField(label='邮箱1')
email2 = serializers.CharField(label='邮箱2', validators=[EmailValidator, ])
email3 = serializers.CharField(label='邮箱3')
email4 = serializers.CharField(label='邮箱4', validators=[MyEmailValidator(r"^w+@\w+\.\w+$")]) def validate_email3(self, value):
"钩子函数"
if not re.match(r"^w+@\w+\.\w+$", value):
raise serializers.ValidationError('格式错误')
return value class User(APIView):
def get(self, request):
return Response({'code':0, 'data':'创建成功'}) def post(self, request):
ser = UserSerializer(data=request.data)
if not ser.is_valid():
return Response({'code':1006, 'data':ser.errors})
print(ser.validated_data)
return Response({'code':0, 'data':'创建成功'})

1.2 ModelSerializer

基于ModelSerializer类,类似Django中的ModelForm

import re
from rest_framework.views import APIView
from rest_framework.views import Response
from rest_framework import serializers class UserModelSerializer(serializers.ModelSerializer): class Meta:
model = UserInfo
# fields = '__all__'
fields = ['username', 'age', 'email', 'password', 'level', 'depart', 'roles'] class User(APIView):
def get(self, request):
return Response({'code': 0, 'data': '创建成功'}) def post(self, request):
ser = UserModelSerializer(data=request.data)
if not ser.is_valid():
return Response({'code': 1006, 'data': ser.errors})
ser.save()
print(ser.validated_data)
return Response({'code': 0, 'data': '创建成功'})

2.序列化

通过ORM从数据库中获取QuerySet或者对象,然后序列化为json格式的数据

模型类

from django.db import models

# Create your models here.
class Role(models.Model):
title = models.CharField(verbose_name='名称', max_length=32) class Department(models.Model):
title = models.CharField(verbose_name='名称', max_length=32) class UserInfo(models.Model):
level_choices = ((1, '普通会员'), (2, 'VIP'), (3, 'SVIP'))
level = models.IntegerField(verbose_name='级别', choices=level_choices, default=1) username = models.CharField(verbose_name='用户名', max_length=32)
password = models.CharField(verbose_name='密码', max_length=32)
age = models.IntegerField(verbose_name='年龄', default=18)
email = models.CharField(verbose_name='邮箱', max_length=64)
token = models.CharField(verbose_name='TOKEN', max_length=64, null=True, blank=True) depart = models.ForeignKey(verbose_name='部门', to='Department', null=True, blank=True, on_delete=models.CASCADE)
roles = models.ManyToManyField(verbose_name='角色', to="Role")

2.1 基本使用

  • 序列化的是一个queryset,则many=True

  • 序列化的是一个对象,则many=False

视图类

from rest_framework.views import APIView
from rest_framework.views import Response
from rest_framework import serializers from web.models import UserInfo class UserModelSerializer(serializers.ModelSerializer):
class Meta:
model = UserInfo
fields = ['username', 'age', 'email'] class User(APIView):
def get(self, request):
"""获取用户列表"""
queryset = UserInfo.objects.all()
ser = UserModelSerializer(instance=queryset, many=True)
return Response({'code': 0, 'data': ser.data})

2.2 自定义字段

  • 将数据库中的choice字段显示中文,如level_text

  • 将ForeignKey字段显示,如depart

  • 自定义字段,如extra,

  • 如果类中自定义的字段名和数据库的一样,则覆盖掉数据库的字段,如depart

extra = serializers.SerializerMethodField()
def get_extra(self,obj):
return 666
 1 from rest_framework.views import APIView
2 from rest_framework.views import Response
3 from rest_framework import serializers
4 from django.core.validators import EmailValidator
5
6 from web.models import UserInfo, Department, Role
7
8
9 class UserModelSerializer(serializers.ModelSerializer):
10 level_text = serializers.CharField(source="get_level_display")
11 depart = serializers.CharField(source="depart.title")
12
13
14 extra = serializers.SerializerMethodField()
15 class Meta:
16 model = UserInfo
17 fields = ['username', 'age', 'email', 'level_text', 'depart', 'extra']
18
19
20 def get_extra(self,obj):
21 return 666
22
23
24 class User(APIView):
25 def get(self, request):
26 """获取用户列表"""
27
28 queryset = UserInfo.objects.all()
29 ser = UserModelSerializer(instance=queryset, many=True)
30 return Response({'code': 0, 'data': ser.data})

2.3 序列化的嵌套

在一个序列化类中,可以嵌套另外一个序列化类

视图

import re
from rest_framework.views import APIView
from rest_framework.views import Response
from rest_framework import serializers
from django.core.validators import EmailValidator from web.models import UserInfo, Department, Role class DepartModelSerializer(serializers.ModelSerializer):
class Meta:
model = Department
fields = '__all__' class RoleModelSerializer(serializers.ModelSerializer):
class Meta:
model = Role
fields = '__all__' class UserModelSerializer(serializers.ModelSerializer):
depart = DepartModelSerializer()
roles = RoleModelSerializer(many=True) class Meta:
model = UserInfo
fields = ['username', 'age', 'email', 'depart', 'roles'] class User(APIView):
def get(self, request):
"""获取用户列表""" queryset = UserInfo.objects.all()
ser = UserModelSerializer(instance=queryset, many=True)
return Response({'code': 0, 'data': ser.data})

3.数据校验&序列化

有时候我们的序列化类可能既要序列化数据,也要校验数据,并且这是常有的,我们就需要给字段设置只读,只写或者可读可写来决定某个字段在序列化或者校验时是否使用本身

  • read_only= True
  • write_only = True
  • 即可读也可写

注意:

如果有嵌套的Serializer,在进行数据校验时,只有两种选择:

1. 将嵌套的序列化设置成 read_only

2. 自定义create和update方法,自定义新建和更新的逻辑

4.源码分析

4.1 类的定义

4.2 序列化

4.3 数据检验

4.4 数据保存

8.drf-序列化器的更多相关文章

  1. drf序列化器serializers.SerializerMethodField()的用法

    问题描述: 为什么DRF中有时候返回的json中图片是带域名的,有时候是不带域名的呢? 解析: 带域名的结果是在view中对模型类序列化的,DRF在序列化图片的时候 会检查上下文有没有request, ...

  2. drf序列化器的实例

    应用目录结构: views.py from django.shortcuts import render # Create your views here. from django.views imp ...

  3. DRF序列化器

    序列化器-Serializer 作用: 1. 序列化,序列化器会把模型对象转换成字典,经过response以后变成json字符串 2. 反序列化,把客户端发送过来的数据,经过request以后变成字典 ...

  4. DRF序列化器的使用

    序列化器的使用 序列化器的使用分两个阶段: 在客户端请求时,使用序列化器可以完成对数据的反序列化. 在服务器响应时,使用序列化器可以完成对数据的序列化. 序列化的基本使用 使用的还是上一篇博文中使用的 ...

  5. DRF 序列化器-Serializer (2)

    作用 1. 序列化,序列化器会把模型对象转换成字典,经过response以后变成json字符串 2. 完成数据校验功能 3. 反序列化,把客户端发送过来的数据,经过request以后变成字典,序列化器 ...

  6. drf序列化器与反序列化

    什么是序列化与反序列化 """ 序列化:对象转换为字符串用于传输 反序列化:字符串转换为对象用于使用 """ drf序列化与反序列化 &qu ...

  7. 对drf序列化器的理解

    序列化: 将对象的状态信息转换为可以存储或传输的形式的过程.(百度定义) 对应到drf中,序列化即把模型对象转换为字典形式, 再返回给前端,主要用于输出 反序列化: 把其他格式转化为程序中的格式. 对 ...

  8. 关于定义序列化器时,read_only和write_only有什么作用

    关于序列化和反序列化 ​ 在谈论前,先说一下序列化和反序列化,这两个概念最初是在学习json的时候提出来的,回头来看,其实可以用最初的理解就可以了 序列化就是将对象转化方便传输和存储字节序列,例如js ...

  9. 一: DRF web应用框架基础,及序列化器的使用

    ---恢复内容开始--- 一: web 应用模式(有两种) 1: 前后端不分离(前端从后端直接获取数据) 2: 前后端分离 二: api 接口 原因一: 为了在团队内部形成共识.防止个人习惯差异引起的 ...

  10. DRF中的序列化器

    DRF中的序列化器详细应用   视图的功能:说白了就是接收前端请求,进行数据处理 (这里的处理包括:如果前端是GET请求,则构造查询集,将结果返回,这个过程为序列化:如果前端是POST请求,假如要对数 ...

随机推荐

  1. laravel框架中验证后在页面提示错误信息

    {{-- 显示错误信息 判断:如果有错误则进行显示,--}} {{-- 通过$errors->any() 获取是否有错误,如果有则返回布尔值true,没有返回布尔值false--}} @if($ ...

  2. 如何实现Windows平台RTMP播放器/RTSP播放器播放窗口添加OSD文字叠加

    好多开发者在做Windows平台特别是单屏多画面显示时,希望像监控摄像机一样,可以在播放画面添加OSD台标,以实现字符叠加效果,大多开发者可很轻松的实现以上效果,针对此,本文以大牛直播SDK (Git ...

  3. 【java8新特性】01:函数式编程及Lambda入门

    我们首先需要先了解什么是函数式编程.函数式编程是一种结构化编程范式.类似于数学函数.它关注的重点在于数据操作.或者说它所提倡的思想是做什么,而不是如何去做. 自Jdk8中开始.它也支持函数式编程.函数 ...

  4. Spring 后置处理器【1】

    Spring 后置处理器[1] 简单介绍 一句话:bean 在初始化前或初始化后的瞬间,我自己添加一些业务逻辑 bean 后置处理器类的内容 简单代码 package com.hspedu.sprin ...

  5. Keepalived+HAProxy 搭建高可用负载均衡

    转载自:https://mp.weixin.qq.com/s/VebiWftaRa26x1aA21Jqww 1. 概述 软件负载均衡技术是指可以为多个后端服务器节点提供前端IP流量分发调度服务的软件技 ...

  6. 通过helm搭建Harbor

    文章转载自:http://www.mydlq.club/article/66/ 系统环境: kubernetes 版本:1.20.1 Traefik Ingress 版本:2.4.3 Harbor C ...

  7. Loki二进制命令帮助

    Usage of config-file-loader: -auth.enabled Set to false to disable auth. (default true) -azure.accou ...

  8. jmeter录制登录脚本

    1.添加代理服务器 在非测试元件添加http代理服务器,端口写8888,域写127.0.0.1 在排除模式里填入.*.(js|css|PNG|jpg|ico|png|gif|woff|ttf).* 2 ...

  9. PAT (Advanced Level) Practice 1002 A+B for Polynomials 分数 25

    This time, you are supposed to find A+B where A and B are two polynomials. Input Specification: Each ...

  10. Vue+vant移动端处理弹窗不能滑动问题

    自己在做项目开发时,使用vantUI组件,在项目中遇到了弹窗组件里面当内容过多时,会出现滚动卡顿或者不能滚动问题,开始一直以为是自己的样式写的有问题,检查下来才发现并不是,而是弹窗组件的问题,于是找到 ...