关于定义序列化器时,read_only和write_only有什么作用
关于序列化和反序列化
在谈论前,先说一下序列化和反序列化,这两个概念最初是在学习json的时候提出来的,回头来看,其实可以用最初的理解就可以了
-
序列化就是将对象转化方便传输和存储字节序列,例如json.dumps就是序列化(狭义的序列化,将字典转化为json字符串),这样得到的json字符串不仅直接可以在其他语言使用(跨平台比较好),而且可以在前后端进行传输交互(drf序列化器)
-
反序列化恰恰相反,而是将字节序列转化为对象,json.loads是将json字符串转化为字典,是狭义的反序列化(因为在python, 一切皆对象,字典是dict( )),而在项目中,前端传过来的序列化数据通过反序列化得到对象,进一步可以通过ORM操作,存入数据库。
关于read_only和write_only(在drf序列化器定义字段时使用)
之所以会有read_only和write_only的疑问,主要和序列化与反序列化有关,所以我会从这两个角度来分析
- read_only 是只读,不能写(可以理解为只能从后台到前台,后台操作后直接传前台),就是不能对字段进行修改操作(也就是不和数据库进行交互),用在序列化字段里。下面的例子中,gender只用于传到前端展示。而不用于存到数据库
#model.py
class User(models.Model):
sex=CharField(
choices=((0, ‘男’), (1,‘女’))
)
#自定义序列化
@property
def gender(self):
return self.get_sex_diaplay()
#serializer.py
class UserModelSerializer(ModelSerializer):
class Meta;
model = models.User
fields = ('gender')
extra_kwargs = {
'gender': {
'read_only':True,
},
}
'''
gender字段只用于把 sex中选项对应的汉字传到前端展示,不与数据库进行交互,而存到数据库的是sex的数字
'''
- write_only是只写不能读,可以理解为只能前台到后台,在后台做了逻辑操作后直接存数据库,在反序列化字段使用。在下面的例子中,password就是反序列化,不需要传前台,而re_password是自定义反序列化字段,仅用作后台和密码进行比较,然后把结果返回前台,所以不存到数据库,在全局钩子使用时要pop掉
#model.py
class User(model.Model):
pwd = models.CharField()
#serializers.py
class UserModelSerializer(ModelSerializer):
re_pwd = serializers.CharField(min_length=3, max_length=64, write_only=True) #自定义反序列化字段,一定是write_only=True
class Meta:
model = models.User
fields = ['pwd', 're_pwd']
extra_kwargs = {
'pwd': {
'write_only': True
}
}
def validate(self, attrs):
pwd = attrs.get('pwd')
re_pwd = attrs.pop('re_pwd')
if pwd != re_pwd:
raise serializers.ValidationError({'re_pwd': '两次密码不一致'})
return attrs