Loading

反序列化(钩子函数进行复杂数据验证)

反序列化(钩子函数进行复杂数据验证)

5.1 验证单个字段

序列化器:

class Student1Serializer1(serializers.Serializer):
    """学生信息序列化器"""

    # 1.转换的字段说明
    # 字段 = serializer.字段类型(选项=选项值)

    id = serializers.IntegerField(read_only=True)  # 反序列化阶段不会要求id有值
    name = serializers.CharField(required=True)  # 反序列化阶段必填
    sex = serializers.CharField(default="男")  # 反序列化阶段如果没填,则使用默认值
    age = serializers.IntegerField(min_value=0, max_value=100, error_messages={
        "min_value": "the age must be age>=100",
        "max_value": "the age must be age<=100"
    })
    classroom = serializers.CharField(default="310",validators=[check_classroom])  # 反序列化阶段如果没填,则使用默认值
    info = serializers.CharField(allow_null=True, allow_blank=True)  # 允许客户端不填写(None),或者为""

    # 2.如果当前序列化器继承的是Modelserializer,则需要声明需要调用的模型信息
    # class Meta:
    #     model = 模型名
    #     fields = ["数据库字段名1","数据库字段名2","数据库字段名3",....,"数据库字段名n",] 或 "__all__"

    # 3.验证代码的对象方法(钩子函数)

    def validate_name(self, data):  # 方法名必须以validate_<字段名>来命名
        """
        验证单个字段
        该方法会自动被is_valid()调用
        """
        if data in ["python","django"]:
            raise serializers.ValidationError("学生姓名不能是python或django")

        # 验证成功必须返回data
        return data

    # 4.模型操作方法

    # def create(self, validated_data): # 添加数据后,字典自动变为模型对象
    #     pass
    #
    # def update(self, instance, validated_data): # 更新数据后,字典自动变为模型对象
    #     pass

视图

class StudentView(View):

    def get(self, request):
        """反序列化-采用字段选项来验证数据-验证失败抛出异常"""
        # 1.接收客户端提交的数据
        # data = json.loads(request.body)
        data = {
            "name": "python",
            "age": 30,
            "sex": "男",
            "classroom": "666",
            "info": "要做个猛男,要猛!"
        }
        # 2.实例化序列化器,获取序列化对象
        serializer = Student1Serializer1(data=data)
        # 3.调用序列化器进行数据验证
        ret = serializer.is_valid() 
        # 4.获取验证以后的结果
        if ret:
            return JsonResponse(dict(serializer.validated_data), safe=False)
        else:
            return JsonResponse(dict(serializer.errors), safe=False)
        # 5.操作数据库

        # 6.返回结果

测试结果

{
    "name": [
        "学生姓名不能是python或django"
    ]
}

5.2 验证多个字段

序列化器:

class Student1Serializer1(serializers.Serializer):
    """学生信息序列化器"""

    # 1.转换的字段说明
    # 字段 = serializer.字段类型(选项=选项值)

    id = serializers.IntegerField(read_only=True)  # 反序列化阶段不会要求id有值
    name = serializers.CharField(required=True)  # 反序列化阶段必填
    sex = serializers.CharField(default="男")  # 反序列化阶段如果没填,则使用默认值
    age = serializers.IntegerField(min_value=0, max_value=100, error_messages={
        "min_value": "the age must be age>=100",
        "max_value": "the age must be age<=100"
    })
    classroom = serializers.CharField(default="310",validators=[check_classroom])  # 反序列化阶段如果没填,则使用默认值
    info = serializers.CharField(allow_null=True, allow_blank=True)  # 允许客户端不填写(None),或者为""

    # 2.如果当前序列化器继承的是Modelserializer,则需要声明需要调用的模型信息
    # class Meta:
    #     model = 模型名
    #     fields = ["数据库字段名1","数据库字段名2","数据库字段名3",....,"数据库字段名n",] 或 "__all__"

    # 3.验证代码的对象方法(钩子函数)
    def validate(self, attrs):  # validate是固定的
        """
        验证来自客户端的所有数据
        :param: attrs 序列化器实例化时,传入的data
        这个方法经常用于密码和确认密码的校验,因为需要多个字段的数据
        """

        # 310教室只有女生,不能有男生
        if attrs["classroom"]=="310" and attrs["sex"] == "男":
            raise serializers.ValidationError("310教室不能有男生,只能有小姐姐。")
        return attrs

    # 4.模型操作方法

    # def create(self, validated_data): # 添加数据后,字典自动变为模型对象
    #     pass
    #
    # def update(self, instance, validated_data): # 更新数据后,字典自动变为模型对象
    #     pass

视图

class StudentView(View):

    def get(self, request):
        """反序列化-采用字段选项来验证数据-验证失败抛出异常"""
        # 1.接收客户端提交的数据
        # data = json.loads(request.body)
        data = {
            "name": "彭于晏",
            "age": 30,
            "sex": "男",
            "classroom": "310",
            "info": "要做个猛男,要猛!"
        }
        # 2.实例化序列化器,获取序列化对象
        serializer = Student1Serializer1(data=data)
        # 3.调用序列化器进行数据验证
        ret = serializer.is_valid() 
        # 4.获取验证以后的结果
        if ret:
            return JsonResponse(dict(serializer.validated_data), safe=False)
        else:
            return JsonResponse(dict(serializer.errors), safe=False)
        # 5.操作数据库

        # 6.返回结果

测试结果

{
    "non_field_errors": [
        "310教室不能有男生,只能有小姐姐。"
    ]
}

5.3 外部函数验证

外部函数

def check_classroom(data):
    """外部验证函数"""
    if data == "444":
        raise serializers.ValidationError("444教室学生已满")

    return data

序列化器:

在要验证的字段的validators里加入外部函数名,注意不要加()调用也不要使用字符串

id = serializers.IntegerField(read_only=True)  # 反序列化阶段不会要求id有值
    name = serializers.CharField(required=True)  # 反序列化阶段必填
    sex = serializers.CharField(default="男")  # 反序列化阶段如果没填,则使用默认值
    age = serializers.IntegerField(min_value=0, max_value=100, error_messages={
        "min_value": "the age must be age>=100",
        "max_value": "the age must be age<=100"
    })
    classroom = serializers.CharField(default="310",validators=[check_classroom])  # 反序列化阶段如果没填,则使用默认值
    info = serializers.CharField(allow_null=True, allow_blank=True)  # 允许客户端不填写(None),或者为""

视图

class StudentView(View):

    def get(self, request):
        """反序列化-采用字段选项来验证数据-验证失败抛出异常"""
        # 1.接收客户端提交的数据
        # data = json.loads(request.body)
        data = {
            "name": "彭于晏",
            "age": 30,
            "sex": "男",
            "classroom": "444",
            "info": "要做个猛男,要猛!"
        }
        # 2.实例化序列化器,获取序列化对象
        serializer = Student1Serializer1(data=data)
        # 3.调用序列化器进行数据验证
        ret = serializer.is_valid() 
        # 4.获取验证以后的结果
        if ret:
            return JsonResponse(dict(serializer.validated_data), safe=False)
        else:
            return JsonResponse(dict(serializer.errors), safe=False)
        # 5.操作数据库

        # 6.返回结果

测试结果

{
    "classroom": [
        "444教室学生已满"
    ]
}

posted @ 2022-10-24 17:13  minqiliang  阅读(43)  评论(0编辑  收藏  举报
-->