7.9.5Serializer序列化器之反序列化操作

7.9.5Serializer序列化器之反序列化操作

(1)调用序列化器进行验证

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

在获取反序列化的数据前,必须调用is_valid()方法进行验证,验证成功返回True,否则返回False。

验证失败,可以通过序列化器对象的errors属性获取错误信息,返回字典,包含了字段和字段的错误。如果是非字段错误,可以通过修改REST framework配置中的NON_FIELD_ERRORS_KEY来控制错误字典中的键名。

验证成功,可以通过序列化器对象的validated_data属性获取数据。

1)准备序列化器

class BookInfoSerializer(serializers.Serializer):

    """图书数据序列化器"""

    id = serializers.IntegerField(label='ID', read_only=True)

    btitle = serializers.CharField(label='名称', max_length=20)

    bpub_date = serializers.DateField(label='发布日期', required=False)

    bread = serializers.IntegerField(label='阅读量', required=False)

    bcomment = serializers.IntegerField(label='评论量', required=False)

    image = serializers.ImageField(label='图片', required=False)

    # 定义一关联多的序列化器字段

    # many=True:表示一对多

    heroinfo_set = serializers.PrimaryKeyRelatedField(read_only=True, many=True)

2)验证失败的情况

>>> from booktest.serializers import BookInfoSerializer

>>> data = {}

>>> s = BookInfoSerializer(data=data)

>>> s.is_valid()

False

>>> s.errors

{'btitle': [ErrorDetail(string='This field is required.', code='required')]}

3)验证成功的情况

>>> data = {'btitle':'水浒传'}

>>> s = BookInfoSerializer(data=data)

>>> s.is_valid()

True

>>> s.validated_data

OrderedDict([('btitle', '水浒传')])

4)验证同时抛出异常

is_valid()方法还可以在验证失败时抛出异常serializers.ValidationError,可以通过传递raise_exception=True参数开启,REST framework接收到此异常,会向前端返回HTTP 400 Bad Request响应。

>>> data = {}

>>> s = BookInfoSerializer(data=data)

>>> s.is_valid(raise_exception=True)

Traceback (most recent call last):

File "", line 1, in

File "/home/python/.virtualenvs/py3_django/lib/python3.5/site- packages/rest_framework/serializers.py", line 244, in is_valid

raise ValidationError(self.errors)

rest_framework.exceptions.ValidationError: {'btitle': [ErrorDetail(string='This field is required.', code='required')]}

(2)定义序列化器进行验证

1)单个字段验证

①在BookInfoSerializer序列化器中添加

def validate_btitle(self,value):

    """单个字段的验证"""

    if 'django' not in value.lower():

        raise serializers.ValidationError("图书不是关于django的")

        return value

②验证

>>> from booktest.serializers import BookInfoSerializer

>>> data = {'btitle':'python'}

>>> s = BookInfoSerializer(data=data)

>>> s.is_valid()

False

>>> s.errors

{'btitle': [ErrorDetail(string='图书不是关于django的', code='invalid')]}

2)对字段联合验证

①提示

多字段联合验证是在单个字段验证之后进行的验证

②在BookInfoSerializer序列化器中添加

def validate(self, attrs):

    """多字段验证"""

    bread = attrs['bread']
    
    bcomment = attrs['bcomment']

    if bread < bcomment:

        raise serializers.ValidationError('阅读量小于评论量')

    return attrs

③验证

>>> from booktest.serializers import BookInfoSerializer

>>> data = {'btitle':'django','bread':10,'bcomment':20}

>>> s = BookInfoSerializer(data=data)

>>> s.is_valid()

False

>>> s.errors

{'non_field_errors': [ErrorDetail(string='阅读量小于评论量', code='invalid')]}

注意:'non_field_errors' 是多字段联合校验时的错误信息key

3)validators

在字段中添加validators选项参数:

btitle = serializers.CharField(label='名称', max_length=20,validators=[about_django])

①在BookInfoSerializer序列化器上方添加(即:定义为全局)

def about_django(value):

    if 'django' not in value.lower(): 

        raise serializers.ValidationError("图书不是关于Django的")

②验证

>>> from booktest.serializers import BookInfoSerializer

>>> data = {'btitle':'python'}

>>> s = BookInfoSerializer(data=data)

>>> s.is_valid()

False

>>> s.errors

{'btitle': [ErrorDetail(string='图书不是关于Django的', code='invalid')]}

(3)保存序列化器验证后的数据

如果在验证成功后,想要基于validated_data完成数据对象的创建,可以通过实现create()和update()两个方法来实现。

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

①在BookInfoSerializer序列化器中添加:

def create(self, validated_data):

    """新建"""

    return BookInfo(**validated_data)



def update(self, instance, validated_data):
    
    """更新,instance为要更新的对象实例"""

    instance.btitle = validated_data.get('btitle', instance.btitle)

    instance.bpub_date = validated_data.get('bpub_date', instance.bpub_date)

    instance.bread = validated_data.get('bread', instance.bread)

    instance.bcomment = validated_data.get('bcomment', instance.bcomment)
    
    return instance

②验证后保存

>>> from booktest.serializers import BookInfoSerializer

>>> data = {'btitle':'django','bread':30,'bcomment':10,'bpub_date':'1989-11-11'}

>>> s = BookInfoSerializer(data=data)

>>> s.is_valid()

True

>>> book = s.save()

>>> book

③验证后新增

>>> data = {'btitle':'python_django','bread':30,'bcomment':10,'bpub_date':'1989-11-11'}

>>> s = BookInfoSerializer(book,data=data)

>>> s.is_valid()

True

>>> book = s.save()

>>> book

两点说明:

1) 在对序列化器进行save()保存时,可以额外传递数据,这些数据可以在create()和update()中的validated_data参数获取到

serializer.save(owner=request.user)

2)默认序列化器必须传递所有required的字段,否则会抛出验证异常。但是我们可以使用partial参数来允许部分字段更新

# Update `comment` with partial data serializer = CommentSerializer(comment, data={'content': u'foo bar'}, partial=True)

 

你可能感兴趣的:(django)