DRF之Serializer序列化组件
目录
序列化类Serializer的使用
serializers序列化组件:
- 序列化过程:可以把我们用ORM产生的QuerySet对象转换成json格式数据;
- 反序列化过程:可以实现数据校验功能,并把客户端发送过来的数据(前端json格式字符串),经过request.data以后变成字典,再把字典转成模型,存到数据库;
简单使用
模型层:
class Book(models.Model):
name = models.CharField(max_length=32)
price = models.IntegerField()
创建序列化类:不想序列化的字段不写就行,建议新建py文件专门存放序列化类。
from rest_framework import serializers
class BookSerializer(serializers.Serializer):
name = serializers.CharField() # 与模型层字段类型对应
price = serializers.IntegerField() # 与模型层字段类型对应
视图层:
class BookAPIView(APIView):
def get(self, request):
# 取出所有图书的queryset对象
books = models.Book.objects.all()
# instance:要序列化的对象
# many:是否有多条数据,只有一条数据就不用传many参数
ser = serializer.BookSerializer(instance=books, many=True)
# 序列化后的数据
print(ser.data)
return Response(ser.data)
序列化类常用字段和字段参数
常见字段
跟模型层的字段一样,但比模型层的字段多几个:
BooleanField()
NullBooleanField()
CharField()
EmailField()
RegexField()
SlugField()
URLField()
UUIDField()
IPAddressField()
IntegerField()
FloatField()
DecimalField()
DateTimeField()
DateField()
TimeField()
ChoiceField()
FileField()
ImageField()
serializer独有字段:
ListField()
DictField()
常见字段参数
serializer中的字段参数是用来反序列化时校验用的。
CharField()
参数 | 作用 |
---|---|
max_length | 最大长度 |
min_lenght | 最小长度 |
allow_blank | 是否允许为空 |
trim_whitespace | 是否截断空白字符 |
IntegerField()
参数 | 作用 |
---|---|
max_value | 最小值 |
min_value | 最大值 |
通用参数
参数 | 作用 |
---|---|
read_only | 表明该字段仅用于序列化输出,默认False |
write_only | 表明该字段仅用于反序列化输入,默认False |
required | 表明该字段在反序列化时必须输入,默认True |
default | 反序列化时使用的默认值 |
allow_null | 表明该字段是否允许传入None,默认False |
error_messages | 包含错误编号与错误信息的字典 |
序列化类Serializer的反序列化
Serializer类还可以帮我们把前端传来的数据进行校验后反序列化,并保存到数据库中。
简单使用
模型层models.py:
class Book(models.Model):
name = models.CharField(max_length=32)
price = models.IntegerField()
Serializer类:
class BookSerializer(serializers.Serializer):
# 校验前端数据时name字段长度不能超过8位
name = serializers.CharField(max_length=8)
price = serializers.IntegerField()
# 新增数据,validated_data是符合条件的字段数据
def create(self, validated_data):
book_obj = models.Book.objects.create(**validated_data)
return book_obj
# 修改数据,validated_data是符合条件的字段数据
# instance是要修改数据的对象
def update(self, instance, validated_data):
# 修改name字段值
instance.name = validated_data.get('name')
# 修改price字段值
instance.price = validated_data.get('price')
instance.save()
return instance
视图层views.py:
class BookAPIView(APIView):
def post(self, request):
# 前端数据传给data参数
ser = BookSerializer(data=request.data)
# 校验数据是否符合规范
if ser.is_valid():
ser.save()
return Response({'msg':'添加成功'})
return Response({'msg':'添加失败'})
def put(self, request, bid):
book_obj = models.Book.objects.filter(pk=bid).first()
ser = BookSerializer(instance=book_obj, data=request.data)
# 校验数据是否符合规范
if ser.is_valid():
ser.save()
return Response({'msg':'修改成功'})
return Response({'msg':'修改失败'})
重点:
- 保存到数据库中用的是Serializer类中的save()方法
- 在使用save()方法前需要先使用is_valid()校验数据
- 添加数据,Serializer类需要自行编写create()方法
- 修改数据,Serializer类需要自行编写update()方法
- 执行save()方法是会判断对象中是否有instance参数,无则执行create()方法,有则执行update()方法
- validated_data是符合条件的字段数据
- create()方法和update()方法需要返回新增的对象和修改后的对象
update()优化写法:如果validated_data没有某个字段,则该字段值用原来的字段值。
def update(self, instance, validated_data):
instance.name = validated_data.get('name', instance.name)
instance.price = validated_data.get('price', instance.price)
instance.save()
return instance
常见方法和属性
方法与属性 | 含义 |
---|---|
save() | 保存到数据库 |
is_valid() | 校验数据是否符合规范 |
errors | 数据校验失败的原因 |
validated_data | 符合条件的数据(只有is_valid()为True才有数据) |
data | 创建对象时传的数据 |
instance | 创建对象时传的模型层对象 |
局部和全局钩子
局部和全局钩子可以再字段校验后进行二次校验。
局部钩子
写在序列化类中的局部钩子函数,校验单个字段。
使用:创建函数名 validate_ + 字段名
from rest_framework import serializers
from rest_framework.serializers import ValidationError
class BookSerializer(serializers.Serializer):
name = serializers.CharField(max_length=8)
price = serializers.IntegerField()
def validate_name(self, name):
"""name参数为校验字段值"""
# 判断name字段值是否已'aa'开头
if name.startwith('aa'):
# 有则抛异常
raise ValidationError('不能已aa开头')
# 符合规范返回字段值
return name
全局钩子
写在序列化类中的全局钩子函数,校验所有字段。
使用:创建函数名 validate
from rest_framework import serializers
from rest_framework.serializers import ValidationError
class BookSerializer(serializers.Serializer):
name = serializers.CharField(max_length=8)
price = serializers.IntegerField()
def validate(self, attrs):
"""attrs参数为校验后的字段数据"""
name=attrs.get('name')
price=attrs.get('price')
# 判断name字段是否等于price字段
if name == price:
# 等于则抛异常
raise ValidationError('名字和价格不能一样')
# 符合规范返回字段数据
return attrs
校验流程:先走字段自己的校验规则,再走局部钩子,再走全局钩子。
序列化类ModelSerializer
ModelSerializer类可以快速完成Serializer类的功能,自动对应模型层的字段。
基本使用
from rest_framework import serializers
class BookSerializer(serializers.ModelSerializer):
# 必要
class Meta:
# 指定模型层中的类
model = Book
# 指定字段
fields = ['name',]
"""
fields = '__all__ 指定所有字段'
"""
# 添加字段校验属性
extra_kwargs = {
'name':{ # 指定name字段长度不能超过8位
'max_length':8,
},
}
也可以把字段单独拿出来:
from rest_framework import serializers
class BookSerializer(serializers.ModelSerializer):
# 必要
class Meta:
...
name = serializers.CharFields(max_length=10)
ModelSerializer类的重点:
- 不需要重写create()和update()方法了
- 局部钩子和全局钩子使用方法和Serializer类一样,注意不要写在Meta类中
- 可以单独重写字段
- 给字段类加属性用extra_kwargs
多表序列化与反序列化
在有外键字段存在时,如果直接把表进行序列化展示,外键字段显示的是外键id值,如果我们想要外键字段的详细信息,并且在添加数据时不会被干扰,可以做如下操作。
现我有如下表:
class Book(models.Model):
name = models.CharField(max_length=32)
price = models.IntegerField()
# 外键字段
publish = models.ForeignKey(to='Publish', on_delete=models.CASCADE)
class Publish(models.Model):
name = models.CharField(max_length=32)
email = models.EmailField()
方法一:在序列化类中编写
ModelSerializer类:
from rest_framework import serializers
from .models import Book
class BookSerializer(serializers.ModelSerializer):
class Meta:
model = Book
fields = [
'name',
'price',
'publish',
'publish_detail',
]
# 设置Book表的publish字段为只用于写入
extra_kwargs = {
'publish':{'write_only':True}
}
# 重写字段,这个字段属性read_only默认为True
# 所以不用添加read_only属性
publish_detail = serializers.SerializerMethodField()
# 配合函数一起使用
def get_publish_detail(self, obj):
"""
obj是Book表对象
返回值:这里返回什么,publish_detail就显示什么
"""
return {
'name': obj.publish.name,
'email': obj.publish.email,
}
方法二:在模型类中编写
模型层:
class Book(models.Model):
name = models.CharField(max_length=32)
price = models.IntegerField()
# 外键字段
publish = models.ForeignKey(to='Publish', on_delete=models.CASCADE)
# 添加一个函数返回publish字段
def publish_detail(self):
return {
'name': self.publish.name,
'email': self.publish.email,
}
ModelSerializer类:
from rest_framework import serializers
from .models import Book
class BookSerializer(serializers.ModelSerializer):
class Meta:
model = Book
fields = [
'name',
'price',
'publish',
'publish_detail', # 这里指的是模型层中的函数
]
# 设置Book表的publish字段为只用于写入
extra_kwargs = {
'publish':{'write_only':True}
}