詳解 DRF 序列化器
一套 RESTful Web API,最基本的要求就是能為調(diào)用該 API 的客戶(hù)端提供符合要求的數(shù)據(jù)類(lèi)型,而能否提供符合要求的數(shù)據(jù)類(lèi)型,序列化器是關(guān)鍵。上一節(jié)中,我們通過(guò)短短 6 行代碼,構(gòu)建了一個(gè)序列化器,那么序列化器是如何工作的呢?這一節(jié),我們就帶領(lǐng)大家手動(dòng)構(gòu)建一個(gè)序列化器,并深入介紹序列化器的詳細(xì)使用方法。
1. 構(gòu)建序列化器
我們?cè)谝延械囊粋€(gè)數(shù)據(jù)庫(kù)模型類(lèi) StudentsInfo:
# models.py
class StudentsInfo(models.Model):
'''
學(xué)生信息模型
'''
s_name = models.CharField(max_length=8, verbose_name='學(xué)生姓名')
s_age = models.IntegerField(verbose_name='學(xué)生年齡')
s_number = models.CharField(max_length=16, verbose_name='學(xué)號(hào)')
構(gòu)建此模型的序列化器,可按如下操作構(gòu)建:
# serializers.py
from rest_framework import serializers
class StudentsSerializer(serializers.Serializer):
'''
學(xué)生信息序列化器
'''
id = serializers.IntegerField(label='ID', read_only=True)
s_name = serializers.CharField(label='姓名', max_length=8)
s_age = serializers.IntegerField(label='年齡', required=True)
s_number = serializers.CharField(label='學(xué)號(hào)', max_length=16)
注意:serializer 不是只能為數(shù)據(jù)庫(kù)模型類(lèi)定義,也可以為非數(shù)據(jù)庫(kù)模型類(lèi)的數(shù)據(jù)定義。serializer 是獨(dú)立于數(shù)據(jù)庫(kù)之外的存在。
2. 常用字段類(lèi)型與約束參數(shù)
常用字段類(lèi)型:
| 字段 | 字段構(gòu)造方式 |
|---|---|
| BooleanField | BooleanField() |
| NullBooleanField | NullBooleanField() |
| CharField | CharField(max_length=None, min_length=None, allow_blank=False, trim_whitespace=True) |
| EmailField | EmailField(max_length=None, min_length=None, allow_blank=False) |
| RegexField | RegexField(regex, max_length=None, min_length=None, allow_blank=False) |
| SlugField | SlugField(max_length=50, minlength=None, allow_blank=False) 正則字段,驗(yàn)證正則模式 [a-zA-Z0-9-]+ |
| URLField | URLField(max_length=200, min_length=None, allow_blank=False) |
| UUIDField | UUIDField(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" |
| IPAddressField | IPAddressField(protocol=‘both’, unpack_ipv4=False, **options) |
| IntegerField | IntegerField(max_value=None, min_value=None) |
| FloatField | FloatField(max_value=None, min_value=None) |
| DecimalField | DecimalField(max_digits, decimal_places, coerce_to_string=None, max_value=None, min_value=None) max_digits: 最多位數(shù) decimal_palces: 小數(shù)點(diǎn)位置 |
| DateTimeField | DateTimeField(format=api_settings.DATETIME_FORMAT, input_formats=None) |
| DateField | DateField(format=api_settings.DATE_FORMAT, input_formats=None) |
| TimeField | TimeField(format=api_settings.TIME_FORMAT, input_formats=None) |
| DurationField | DurationField() |
| ChoiceField | ChoiceField(choices) choices與Django的用法相同 |
| MultipleChoiceField | MultipleChoiceField(choices) |
| FileField | FileField(max_length=None, allow_empty_file=False, use_url=UPLOADED_FILES_USE_URL) |
| ImageField | ImageField(max_length=None, allow_empty_file=False, use_url=UPLOADED_FILES_USE_URL) |
| ListField | ListField(child=, min_length=None, max_length=None) |
| DictField | DictField(child=) |
選項(xiàng)參數(shù):
| 參數(shù)名稱(chēng) | 作用 |
|---|---|
| max_length | 最大長(zhǎng)度 |
| min_lenght | 最小長(zhǎng)度 |
| allow_blank | 是否允許為空 |
| trim_whitespace | 是否截?cái)嗫瞻鬃址?/td> |
| max_value | 最小值 |
| min_value | 最大值 |
通用參數(shù):
| 參數(shù)名稱(chēng) | 說(shuō)明 |
|---|---|
| read_only | 表明該字段僅用于序列化輸出,默認(rèn)False |
| write_only | 表明該字段僅用于反序列化輸入,默認(rèn)False |
| required | 表明該字段在反序列化時(shí)必須輸入,默認(rèn)True |
| default | 反序列化時(shí)使用的默認(rèn)值 |
| allow_null | 表明該字段是否允許傳入None,默認(rèn)False |
| validators | 該字段使用的驗(yàn)證器 |
| error_messages | 包含錯(cuò)誤編號(hào)與錯(cuò)誤信息的字典 |
| label | 用于HTML展示API頁(yè)面時(shí),顯示的字段名稱(chēng) |
| help_text | 用于HTML展示API頁(yè)面時(shí),顯示的字段幫助提示信息 |
3. Serializer對(duì)象
定義好 Serializer 類(lèi)后,就可以創(chuàng)建 Serializer 對(duì)象了。Serializer 的構(gòu)造方法為:
Serializer(instance=None, data=empty, **kwarg)
說(shuō)明:
-
用于序列化時(shí),將模型類(lèi)對(duì)象傳入 instance 參數(shù);
-
用于反序列化時(shí),將要被反序列化的數(shù)據(jù)傳入 data 參數(shù);
-
除了 instance 和 data 參數(shù)外,在構(gòu)造 Serializer 對(duì)象時(shí),還可通過(guò) context 參數(shù)額外添加數(shù)據(jù),如:
serializer = AccountSerializer(account, context={'request': request})
通過(guò) context 參數(shù)附加的數(shù)據(jù),可以通過(guò) Serializer 對(duì)象的 context 屬性獲取。
4. 序列化器使用流程
為了更好理解序列化器的原理,我們通過(guò)終端來(lái)演示序列化器的使用:
python manage.py shell
- 通過(guò)查詢(xún)獲取需要序列化的學(xué)生對(duì)象:
from AppDemo.models import StudentsModel
student = StudentsModel.objects.get(pk=2)
- 實(shí)例化序列化器,構(gòu)造序列化器對(duì)象:
from AppDemo.serializers import StudentsSerializer
serializer = StudentsSerializer(student)
- 獲取序列化數(shù)據(jù),通過(guò)序列化器對(duì)象的 data 屬性,可以獲取序列化后的數(shù)據(jù):
serializer.data
# {'id': 1, 's_name': '小明', 's_age': 12, 's_number': '001'}
- 如果要被序列化的是包含多條數(shù)據(jù)的查詢(xún)集 QuerySet,可以通過(guò)添加 many=True 參數(shù)實(shí)現(xiàn)序列化
student_qs = StudentsModel.objects.all()
serializer = StudentsSerializer(student_qs, many=True)
serializer.data
# [OrderedDict([('id', 1), ('s_name', '小明'), ('s_age', 12), ('s_number', '001')]), OrderedDict([('id', 3), ('s_name', '小王'), ('s_age', 11), ('s_number', '003')]), OrderedDict([('id', 4), ('s_name', '小白'), ('s_age', 10), ('s_number', '004')])]
5.反序列化使用
5.1 數(shù)據(jù)驗(yàn)證
序列化是將數(shù)據(jù)庫(kù)中的數(shù)據(jù)進(jìn)行解析,返回前端需要的數(shù)據(jù)形式(如 json),而前端發(fā)來(lái)的數(shù)據(jù),經(jīng)過(guò)驗(yàn)證和處理,變?yōu)閿?shù)據(jù)庫(kù)需要的數(shù)據(jù)形式,這個(gè)過(guò)程稱(chēng)為反序列化。
為了保證數(shù)據(jù)符合數(shù)據(jù)模型的要求,對(duì)前端發(fā)來(lái)的數(shù)據(jù),根據(jù)字段類(lèi)別、字段限制首先進(jìn)行驗(yàn)證。在序列化器中,可以調(diào)用 is_valid() 方法進(jìn)行驗(yàn)證,驗(yàn)證成功返回 True,否則返回 False。
# serializers.py
from rest_framework import serializers
class StudentsSerializer(serializers.Serializer):
id = serializers.IntegerField(label='ID', read_only=True)
s_name = serializers.CharField(label='姓名', max_length=8)
s_age = serializers.IntegerField(label='年齡', required=True)
s_number = serializers.CharField(label='學(xué)號(hào)', max_length=16)
通過(guò)構(gòu)造序列化器對(duì)象,并將要反序列化的數(shù)據(jù)傳遞給 data 構(gòu)造參數(shù),進(jìn)而進(jìn)行驗(yàn)證:
from AppDemo.serializers import StudentsSerializer
data = {'s_name': '小黃', 's_age': 15, 's_number': '004'}
serializer = StudentsSerializer(data=data)
serializer.is_valid()
# 返回True
serializer.errors
# {}
serializer.validated_data # 驗(yàn)證通過(guò)的數(shù)據(jù)
# OrderedDict([('s_name', '小黃'), ('s_age', 15), ('s_number', '004')])
5.2 數(shù)據(jù)保存
數(shù)據(jù)驗(yàn)證通過(guò)后,如果需要將驗(yàn)證通過(guò)的數(shù)據(jù)(validated_data)進(jìn)行存儲(chǔ)或更新,需要對(duì)序列化器進(jìn)行改造,增加 create() 和 update() 兩個(gè)方法來(lái)實(shí)現(xiàn):
from rest_framework import serializers
from AppDemo.models import StudentsModel
class StudentsSerializer(serializers.Serializer):
id = serializers.IntegerField(label='ID', read_only=True)
s_name = serializers.CharField(label='姓名', max_length=8)
s_age = serializers.IntegerField(label='年齡', required=True)
s_number = serializers.CharField(label='學(xué)號(hào)', max_length=16)
def create(self, validated_data):
"""存儲(chǔ)新數(shù)據(jù)"""
return StudentsModel.objects.create(**validated_data)
def update(self, instance, validated_data):
"""更新,instance為要更新的對(duì)象實(shí)例"""
instance.s_name = validated_data.get('s_name', instance.s_name)
instance.s_age = validated_data.get('s_age', instance.s_age)
instance.s_number = validated_data.get('s_number', instance.s_number)
instance.save()
return instance
實(shí)現(xiàn)了上述兩個(gè)方法后,在反序列化數(shù)據(jù)的時(shí)候,就可以通過(guò) save() 方法返回一個(gè)數(shù)據(jù)對(duì)象實(shí)例了
student = serializer.save()
說(shuō)明:
在對(duì)序列化器進(jìn)行 save() 保存時(shí),可以額外傳遞數(shù)據(jù),這些數(shù)據(jù)可以在 create() 和 update() 中的validated_data 參數(shù)獲取到:
serializer.save(owner=request.user)
默認(rèn)序列化器必須傳遞所有 required 的字段,否則會(huì)拋出驗(yàn)證異常。但是我們可以使用 partial 參數(shù)來(lái)允許部分字段更新:
serializer = CommentSerializer(comment, data={'content': u'foo bar'}, partial=True)
6. 模型類(lèi)序列化器 ModelSerializer
在上述案例中,我們手動(dòng)實(shí)現(xiàn)了序列化器。在實(shí)現(xiàn)的過(guò)程中,每一個(gè)字段都需要指定,如果模型類(lèi)中字段很多,那將是一個(gè)極其繁瑣的過(guò)程。能否通過(guò)模型中已聲明的字段,自動(dòng)創(chuàng)建序列化器,從而簡(jiǎn)化操作呢?幸運(yùn)的是,Django REST framework 為我們提供了 ModelSerializer 模型類(lèi)序列化器。
ModelSerializer 與常規(guī)的 Serializer 相同,不再需要聲明每一個(gè)字段,它提供了:
- 基于模型類(lèi)自動(dòng)生成一系列字段
- 包含默認(rèn)的 create() 和 update() 的實(shí)現(xiàn)
6.1 構(gòu)建模型類(lèi)序列化器(ModelSerializer)
將上述序列化器改造為模型類(lèi)序列化器 StudentsSerializer:
# serializers.py
from rest_framework import serializers
from AppDemo.models import StudentsModel
class StudentsSerializer(serializers.ModelSerializer):
class Meta:
model = StudentsModel
fields = '__all__'
-
model 指明與序列化器相對(duì)應(yīng)的數(shù)據(jù)模型;
-
fields 指明為模型類(lèi)中要序列化(反序列化)的字段。
我們?cè)诮K端中可以查看我們構(gòu)建的模型類(lèi)序列化器:
python manage.py shell
可以看到,模型類(lèi)序列化器與我們手動(dòng)實(shí)現(xiàn)的序列化器并無(wú)差別,簡(jiǎn)化的工作,只是 Django REST framework 幫助我們實(shí)現(xiàn)了。
from AppDemo.serializers import StudentsSerializer
serializer = StudentsSerializer()
serializer
# StudentsSerializer():
id = IntegerField(label='ID', read_only=True)
s_name = CharField(label='學(xué)生姓名', max_length=8)
s_age = IntegerField(label='學(xué)生年齡')
s_number = CharField(label='學(xué)號(hào)', max_length=16)
6.2 指定字段
- 在構(gòu)建序列化器時(shí),fields 除了可以指定
__all__外,還可以選擇性地指定需要序列化的字段。
# serializers.py
from rest_framework import serializers
from AppDemo.models import StudentsModel
class StudentsSerializer(serializers.ModelSerializer):
class Meta:
model = StudentsModel
fields = ('id', 's_name', 's_age', 's_number')
- 另外,也可以使用 exclude 用來(lái)指定哪些字段不需要進(jìn)行序列化。
class BookInfoSerializer(serializers.ModelSerializer):
"""圖書(shū)數(shù)據(jù)序列化器"""
class Meta:
model = BookInfo
exclude = ('image',)
- 指明只讀字段,有些字段僅需序列化時(shí)輸出,此時(shí)可以通過(guò)read_only_fields指明只讀字段:
# serializers.py
from rest_framework import serializers
from AppDemo.models import StudentsModel
class StudentsSerializer(serializers.ModelSerializer):
class Meta:
model = StudentsModel
fields = ('id', 's_name', 's_age', 's_number')
read_only_fields = ('id')
7.小結(jié)
本節(jié)小節(jié)通過(guò)手動(dòng)構(gòu)建序列化器的方式,帶領(lǐng)大家詳細(xì)了解了序列化器的構(gòu)建過(guò)程和使用流程,并帶領(lǐng)大家認(rèn)識(shí)了更為簡(jiǎn)潔的模型類(lèi)序列化器。在實(shí)際工作中,90% 以上的場(chǎng)景可以直接使用模型類(lèi)序列化器實(shí)現(xiàn),使用模型類(lèi)序列化器能夠大大減少代碼量,讓程序更加簡(jiǎn)潔,同時(shí)提高開(kāi)發(fā)效率。
ExcellentDavid ·
2025 imooc.com All Rights Reserved |