4.序列化器-Serializer

序列化器-Serializer

作用:

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

1. 定义序列化器

models.py

from django.db import models

# Create your models here.
class Student(models.Model):
    # 模型字段
    name = models.CharField(max_length=100,verbose_name="姓名",help_text="提示文本:账号不能为空!")
    sex = models.BooleanField(default=True,verbose_name="性别")
    age = models.IntegerField(verbose_name="年龄")
    class_null = models.CharField(max_length=5,verbose_name="班级编号")
    description = models.TextField(verbose_name="个性签名")

    class Meta:
        db_table="tb_student"
        verbose_name = "学生"
        verbose_name_plural = verbose_name

serializers.py

from rest_framework import serializers

#声明序列化器,所有的序列化器都要直接或者间接继承于 Serializer
class StudentSerializer(serializers.Serializer):
    """学生信息序列化器"""
    # 1. 需要进行数据转换的字段
    id = serializers.IntegerField()
    name = serializers.CharField()
    age = serializers.IntegerField()
    sex = serializers.BooleanField()
    description = serializers.CharField()
    
    # 2.编写添加和更新模型的代码
    def create(self, validated_data):
        pass
    def update(self, instance, validated_data):
        pass
    # 3. 验证代码(反序列化时使用)

    

注意:

1.serializer不是只能为数据库模型类定义,也可以为非数据库模型类的数据定义.

2.创建Serializer对象

Serializer的构造方法为:

Serializer(instance=None, data=empty, **kwarg)

说明:

参数名含义
instance用于序列化时,将模型类对象传入
data用于反序列化时,将要被反序列化的数据传入
context额外添加数据serializer = AccountSerializer(account, context={'request': request})通过context参数附加的数据,可以通过Serializer对象的context属性获取。
many当instance参数为模型列表则many的值必须是True
  1. 序列化器需要在视图中手动调用
  2. 使用序列化器时需要把数据传递过来
  3. 序列化器的字段声明类表单系统。

3.序列化器的使用

序列化器的使用分两个阶段:

  1. 在客户端请求时,使用序列化器可以完成对数据的反序列化。
  2. 在服务器响应时,使用序列化器可以完成对数据的序列化。

3.1 序列化

1) 先查询出一个学生对象

from students.models import Student
student = Student.objects.get(id=3)

2) 构造序列化器对象

from .serializers import StudentSerializer
serializer = StudentSerializer(instance=student)

3)获取序列化数据

通过data属性可以获取序列化后的数据

serializer.data
# {'id': 4, 'name': '小张', 'age': 18, 'sex': True, 'description': '猴赛雷'}

4)完整视图代码:

urls.py

from . import views
urlpatterns = [
    re_path("student/(?P<pk>[0-9]{4})", views.StudentView.as_view())
]

views.py

from django.views import View
from students.models import Student
from .serializers import StudentSerializer
from django.http.response import JsonResponse
class StudentView(View):
    """使用序列化器序列化转换单个模型数据"""
    def get(self,request,pk):
        # 获取数据
        student = Student.objects.get(pk=pk)
        # 数据转换[序列化过程]
        serializer = StudentSerializer(instance=student)
        print(serializer.data)
        # 响应数据
        return JsonResponse(serializer.data)

5)如果要被序列化的是包含多条数据的查询集QuerySet,可以通过添加many=True参数补充说明

urls.py

from . import views
urlpatterns = [
    path("student/", views.StudentView.as_view())
]

view.py

    """使用序列化器序列化转换多个模型数据"""
    def get(self,request):
        # 获取数据
        student_list = Student.objects.all()

        # 转换数据[序列化过程]
        # 如果转换多个模型对象数据,则需要加上many=True
        serializer = StudentSerializer(instance=student_list,many=True)
        print( serializer.data ) # 序列化器转换后的数据

        # 响应数据给客户端
        # 返回的json数据,如果是列表,则需要声明safe=False
        return JsonResponse(serializer.data,safe=False)
    
    
    # 访问结果:
    # [OrderedDict([('id', 1), ('name', 'xiaoming'), ('age', 20), ('sex', True), ('description', '测试')]), OrderedDict([('id', 2), ('name', 'xiaohui'), ('age', 22), ('sex', True), ('description', '后面来的测试')]), OrderedDict([('id', 4), ('name', '小张'), ('age', 18), ('sex', True), ('description', '猴赛雷')])]

3.2 反序列化

3.2.1 数据验证

使用序列化器进行反序列化时,需要对数据进行验证后,才能获取验证成功的数据或保存成模型类对象。

mysite/urls.py

path("drf/", include("students.urls")),

students/urls

from django.urls import path
from . import views

urlpatterns = [
    path("student/", views.StudentView.as_view())
]

students/serializers.py

#反序列验证方法三:
def check_sex_value(data):
    # 字段验证函数
    if data not in [1, 0]:
        raise serializers.ValidationError("sex value only have 1 and 0")


class StudentSerializer(serializers.Serializer):
    id = serializers.IntegerField(label="ID", read_only=True)
    name = serializers.CharField(label="名称", max_length=20, required=True)
    #反序列验证方法三
    sex = serializers.IntegerField(label="性别", required=False, allow_null=True, validators=[check_sex_value])
    age = serializers.IntegerField(label="年龄", required=True)
    class_num = serializers.CharField(label="班级编号", required=False)
    description = serializers.CharField(required=False, allow_null=True, allow_blank=True)
	
    #反序列验证方法一:
    def validate_name(self, attr):
        print(f"attr:{attr}")
        if attr == "老鹰":
            raise serializers.ValidationError("对不起你太凶了")
        return attr
	#反序列验证方法二:
    def validate(self, attrs):
        print(attrs)
        if attrs.get("age") > 60 and attrs.get("sex") == 1:
            raise serializers.ValidationError("对不起男人的年龄不能超过60")
        return attrs

students/views.py

class StudentView(View):
    def get(self, request):
        student = Student.objects.all()
        #通过构造序列化器对象,并将要反序列化的数据传递给data构造参数,进而进行验证
        serializer = StudentSerializer(instance=student, many=True)
        print(serializer.data)
        return JsonResponse(serializer.data, safe=False)

    def post(self, request):
        print(request.body)
        data = json.loads(request.body)
        serializer = StudentSerializer(data=data)
        #raise_exception=True验证失败时抛出异常serializers.ValidationError
        #会向前端返回HTTP 400 Bad Request响应
        result=serializer.is_valid(raise_exception=True)
        print(f"成功:{result}")
        print(f"成功数据:{serializer.validated_data()}")
        print(f"失败:{serializer.errors}")
        return HttpResponse("ok")
12
反序列验证方法一validate_字段名,对<field_name>字段进行验证
反序列验证方法一validate,同时对多个字段进行比较验证时
反序列验证方法三validators在字段中添加validators选项参数,补充验证行为
3.2.2 数据保存和更新

前面的验证数据成功后,我们可以使用序列化器来完成数据反序列化的过程.这个过程可以把数据转成模型类对象.

可以通过实现create()和update()两个方法来实现。

students/serializer.py

class StudentSerializer(serializers.Serializer):
    id = serializers.IntegerField(label="ID", read_only=True)
    name = serializers.CharField(label="名称", max_length=20, required=True)
    sex = serializers.IntegerField(label="性别", required=False, allow_null=True, validators=[check_sex_value])
    age = serializers.IntegerField(label="年龄", required=True)
    class_num = serializers.CharField(label="班级编号", required=False)
    description = serializers.CharField(required=False, allow_null=True, allow_blank=True)

    @staticmethod
    def validate_name(attr):
        if attr == "老鹰":
            raise serializers.ValidationError("对不起你太凶了")
        return attr

    def validate(self, attrs):
        if attrs.get("age") > 60 and attrs.get("sex") == 1:
            raise serializers.ValidationError("对不起男人的年龄不能超过60")
        return attrs

    def create(self, validated_data):
        return Student.objects.create(**validated_data)

    def update(self, instance, validated_data):
        instance.name = validated_data.get("name", instance.name)
        instance.sex = validated_data.get("sex", instance.sex)
        instance.age = validated_data.get("age", instance.age)
        instance.class_num = validated_data.get("class_num", instance.class_num)
        instance.description = validated_data.get("description", instance.description)
        instance.save()
        return instance

students/views.py

class StudentView(View):
    def get(self, request):
        student = Student.objects.all()
        serializer = StudentSerializer(instance=student, many=True)
        print(serializer.data)
        return JsonResponse(serializer.data, safe=False)

    def post(self, request):
        print(request.body)
        data = json.loads(request.body)
        serializer = StudentSerializer(data=data)
        print(f"成功:{serializer.is_valid(raise_exception=True)}")
        print(f"失败:{serializer.errors}")
        serializer.save()
        return HttpResponse("生成新数据成功")

    def put(self, request):
        print(request.body)
        data = json.loads(request.body)
        put_id = data.get("id")
        student = Student.objects.get(pk=put_id)
        # 使用partial参数来允许部分字段更新,即使存在required字段没有提交也不会抛出异常
        serializer = StudentSerializer(instance=student, data=data, partial=True)
        serializer.is_valid(raise_exception=True)
        # save中传递的关键字参数在create,和update方法中也可以获取
        data=serializer.save()#save()方法可以返回一个数据对象实例
        return HttpResponse("更新完成")

实现了上述两个方法后,在反序列化数据的时候,就可以通过save()方法返回一个数据对象实例了

如果创建序列化器对象的时候,没有传递instance实例,则调用save()方法的时候,create()被调用,相反,如果传递了instance实例,则调用save()方法的时候,update()被调用。

3.3 模型类序列化器

如果想为Django的模型类提供序列化器可以使用ModelSerializer模型类快速创建一个Serializer类

ModelSerializer与常规的Serializer相同,但提供了

  • 基于模型类自动生成一系列字段
  • 基于模型类自动为Serializer生成validators,比如unique_together
  • 包含默认的create()和update()的实现
class StudentModel2Serializer(serializers.ModelSerializer):
    class Meta:
        model = Student
        fields = ("id", "name", "sex", "age", "class_num")
        exclude = ("description",)
        read_only_fields = ('id',)
        extra_kwargs = {
            "id": { "required": True},
            "name": {"required": True},
            "age": {"max_value": 60, "required": True}
        }
class Student2ViewSet(View):
    def get(self):
        student = Student.objects.all()
        serializer = StudentModel2Serializer(instance=student, many=True)
        print(serializer.data)
        return JsonResponse(serializer.data, safe=False)

    def post(self, request):
        print(request.body)
        data = json.loads(request.body)
        serializer = StudentModel2Serializer(data=data)
        serializer.is_valid(raise_exception=True)
        serializer.save()
        return HttpResponse(serializer.data)

字段说明

字段名含义
model指明需要声明调用的模型类
fields指明为模型类的哪些字段生成__all__表名包含所有字段,也可以写明具体哪些字段
exclude明确排除掉哪些字段
read_only_fields指明只读字段,即仅用于序列化输出
extra_kwarg为ModelSerializer添加或修改原有的选项参数

3.4 序列化器参考

1.常用字段类型:
字段字段构造方式
BooleanFieldBooleanField()
NullBooleanFieldNullBooleanField()
CharFieldCharField(max_length=None, min_length=None, allow_blank=False, trim_whitespace=True)
EmailFieldEmailField(max_length=None, min_length=None, allow_blank=False)
RegexFieldRegexField(regex, max_length=None, min_length=None, allow_blank=False)
SlugFieldSlugField(maxlength=50, min_length=None, allow_blank=False) 正则字段,验证正则模式 [a-zA-Z0-9-]+
URLFieldURLField(max_length=200, min_length=None, allow_blank=False)
UUIDFieldUUIDField(format=‘hex_verbose’) format: 1) 'hex_verbose'"5ce0e9a5-5ffa-654b-cee0-1238041fb31a" 2) 'hex'"5ce0e9a55ffa654bcee01238041fb31a" 3)'int' - 如: "123456789012312313134124512351145145114" 4)'urn' 如: "urn:uuid:5ce0e9a5-5ffa-654b-cee0-1238041fb31a"
IPAddressFieldIPAddressField(protocol=‘both’, unpack_ipv4=False, **options)
IntegerFieldIntegerField(max_value=None, min_value=None)
FloatFieldFloatField(max_value=None, min_value=None)
DecimalFieldDecimalField(max_digits, decimal_places, coerce_to_string=None, max_value=None, min_value=None) max_digits: 最多位数 decimal_palces: 小数点位置
DateTimeFieldDateTimeField(format=api_settings.DATETIME_FORMAT, input_formats=None)
DateFieldDateField(format=api_settings.DATE_FORMAT, input_formats=None)
TimeFieldTimeField(format=api_settings.TIME_FORMAT, input_formats=None)
DurationFieldDurationField()
ChoiceFieldChoiceField(choices) choices与Django的用法相同
MultipleChoiceFieldMultipleChoiceField(choices)
FileFieldFileField(max_length=None, allow_empty_file=False, use_url=UPLOADED_FILES_USE_URL)
ImageFieldImageField(max_length=None, allow_empty_file=False, use_url=UPLOADED_FILES_USE_URL)
ListFieldListField(child=, min_length=None, max_length=None)
DictFieldDictField(child=)
2.选项参数
参数名称作用
max_length最大长度
min_lenght最小长度
allow_blank是否允许为空
trim_whitespace是否截断空白字符
max_value最小值
min_value最大值
3.通用参数
参数名称说明
read_only表明该字段仅用于序列化输出,默认False
write_only表明该字段仅用于反序列化输入,默认False
required表明该字段在反序列化时必须输入,默认True
default反序列化时使用的默认值
allow_null表明该字段是否允许传入None,默认False
validators该字段使用的验证器
error_messages包含错误编号与错误信息的字典
label用于HTML展示API页面时,显示的字段名称
help_text用于HTML展示API页面时,显示的字段帮助提示信息
  • 2
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值