DRF的序列化
为什么要用序列化组件
当我们做前后端分离的项目~~我们前后端交互一般都选择JSON数据格式,JSON是一个轻量级的数据交互格式。
那么我们给前端数据的时候都要转成json格式,那就需要对我们从数据库拿到的数据进行序列化。
接下来我们看下django序列化和rest_framework序列化的对比~~
Django的序列化方法
1 class BooksView(View): 2 def get(self, request): 3 book_list = Book.objects.values("id", "title", "chapter", "pub_time", "publisher") 4 book_list = list(book_list) 5 # 如果我们需要取外键关联的字段信息 需要循环获取外键 再去数据库查然后拼接成我们想要的 6 ret = [] 7 for book in book_list: 8 pub_dict = {} 9 pub_obj = Publish.objects.filter(pk=book["publisher"]).first() 10 pub_dict["id"] = pub_obj.pk 11 pub_dict["title"] = pub_obj.title 12 book["publisher"] = pub_dict 13 ret.append(book) 14 ret = json.dumps(book_list, ensure_ascii=False, cls=MyJson) 15 return HttpResponse(ret) 16 17 18 # json.JSONEncoder.default() 19 # 解决json不能序列化时间字段的问题 20 class MyJson(json.JSONEncoder): 21 def default(self, field): 22 if isinstance(field, datetime.datetime): 23 return field.strftime('%Y-%m-%d %H:%M:%S') 24 elif isinstance(field, datetime.date): 25 return field.strftime('%Y-%m-%d') 26 else: 27 return json.JSONEncoder.default(self, field)
1 from django.core import serializers 2 3 4 # 能够得到我们要的效果 结构有点复杂 5 class BooksView(View): 6 def get(self, request): 7 book_list = Book.objects.all() 8 ret = serializers.serialize("json", book_list) 9 return HttpResponse(ret)
DRF序列化的方法
首先,我们要用DRF的序列化,就要遵循人家框架的一些标准,
-- Django我们CBV继承类是View,现在DRF我们要用APIView
-- Django中返回的时候我们用HTTPResponse,JsonResponse,render ,DRF我们用Response
为什么这么用~我们之后会详细讲~~我们继续来看序列化~~
序列化
1 class BookSerializer(serializers.Serializer): 2 id = serializers.IntegerField() 3 title = serializers.CharField(max_length=32) 4 CHOICES = ((1, "Linux"), (2, "Django"), (3, "Python")) 5 chapter = serializers.ChoiceField(choices=CHOICES, source="get_chapter_display") 6 pub_time = serializers.DateField()
1 from rest_framework.views import APIView 2 from rest_framework.response import Response 3 4 class BookView(APIView): 5 def get(self, request): 6 book_list = Book.objects.all() 7 ret = BookSerializer(book_list, many=True) 8 return Response(ret.data)
外键关系的序列化
1 # by gaoxin 2 from rest_framework import serializers 3 from .models import Book 4 5 6 class PublisherSerializer(serializers.Serializer): 7 id = serializers.IntegerField(read_only=True) 8 title = serializers.CharField(max_length=32) 9 10 11 class UserSerializer(serializers.Serializer): 12 id = serializers.IntegerField(read_only=True) 13 name = serializers.CharField(max_length=32) 14 age = serializers.IntegerField() 15 16 17 class BookSerializer(serializers.Serializer): 18 id = serializers.IntegerField(read_only=True) 19 title = serializers.CharField(max_length=32) 20 CHOICES = ((1, "Linux"), (2, "Django"), (3, "Python")) 21 chapter = serializers.ChoiceField(choices=CHOICES, source="get_chapter_display", read_only=True) 22 pub_time = serializers.DateField() 23 24 publisher = PublisherSerializer(read_only=True) 25 user = UserSerializer(many=True, read_only=True)
反序列化
当前端给我们发post的请求的时候~前端给我们传过来的数据~我们要进行一些校验然后保存到数据库~
这些校验以及保存工作,DRF的Serializer也给我们提供了一些方法了~~
首先~我们要写反序列化用的一些字段~有些字段要跟序列化区分开~~
Serializer提供了.is_valid() 和.save()方法~~
1 # serializers.py 文件 2 class BookSerializer(serializers.Serializer): 3 id = serializers.IntegerField(read_only=True) 4 title = serializers.CharField(max_length=32) 5 CHOICES = ((1, "Linux"), (2, "Django"), (3, "Python")) 6 chapter = serializers.ChoiceField(choices=CHOICES, source="get_chapter_display", read_only=True) 7 w_chapter = serializers.IntegerField(write_only=True) 8 pub_time = serializers.DateField() 9 10 publisher = PublisherSerializer(read_only=True) 11 user = UserSerializer(many=True, read_only=True) 12 13 users = serializers.ListField(write_only=True) 14 publisher_id = serializers.IntegerField(write_only=True) 15 16 def create(self, validated_data): 17 book = Book.objects.create(title=validated_data["title"], chapter=validated_data["w_chapter"], pub_time=validated_data["pub_time"], publisher_id=validated_data["publisher_id"]) 18 book.user.add(*validated_data["users"]) 19 return book
1 class BookView(APIView): 2 def get(self, request): 3 book_list = Book.objects.all() 4 ret = BookSerializer(book_list, many=True) 5 return Response(ret.data) 6 7 def post(self, request): 8 # book_obj = request.data 9 print(request.data) 10 serializer = BookSerializer(data=request.data) 11 if serializer.is_valid(): 12 print(12341253) 13 serializer.save() 14 return Response(serializer.validated_data) 15 else: 16 return Response(serializer.errors)
当前端给我们发送patch请求的时候,前端传给我们用户要更新的数据,我们要对数据进行部分验证~~
1 class BookSerializer(serializers.Serializer): 2 id = serializers.IntegerField(read_only=True) 3 title = serializers.CharField(max_length=32) 4 CHOICES = ((1, "Linux"), (2, "Django"), (3, "Python")) 5 chapter = serializers.ChoiceField(choices=CHOICES, source="get_chapter_display", read_only=True) 6 w_chapter = serializers.IntegerField(write_only=True) 7 pub_time = serializers.DateField() 8 9 publisher = PublisherSerializer(read_only=True) 10 user = UserSerializer(many=True, read_only=True) 11 12 users = serializers.ListField(write_only=True) 13 publisher_id = serializers.IntegerField(write_only=True) 14 15 def create(self, validated_data): 16 book = Book.objects.create(title=validated_data["title"], chapter=validated_data["w_chapter"], pub_time=validated_data["pub_time"], 17 publisher_id=validated_data["publisher_id"]) 18 book.user.add(*validated_data["users"]) 19 return book 20 21 def update(self, instance, validated_data): 22 instance.title = validated_data.get("title", instance.title) 23 instance.chapter = validated_data.get("w_chapter", instance.chapter) 24 instance.pub_time = validated_data.get("pub_time", instance.pub_time) 25 instance.publisher_id = validated_data.get("publisher_id", instance.publisher_id) 26 if validated_data.get("users"): 27 instance.user.set(validated_data.get("users")) 28 instance.save() 29 return instance
1 class BookView(APIView): 2 def patch(self, request): 3 print(request.data) 4 book_id = request.data["id"] 5 book_info = request.data["book_info"] 6 book_obj = Book.objects.filter(pk=book_id).first() 7 serializer = BookSerializer(book_obj, data=book_info, partial=True) 8 if serializer.is_valid(): 9 serializer.save() 10 return Response(serializer.validated_data) 11 else: 12 return Response(serializer.errors)
验证
如果我们需要对一些字段进行自定义的验证~DRF也给我们提供了钩子方法~~
1 class BookSerializer(serializers.Serializer): 2 id = serializers.IntegerField(read_only=True) 3 title = serializers.CharField(max_length=32) 4 # 省略了一些字段 跟上面代码里一样的 5 # 。。。。。 6 def validate_title(self, value): 7 if "python" not in value.lower(): 8 raise serializers.ValidationError("标题必须含有Python") 9 return value
1 class BookSerializer(serializers.Serializer): 2 id = serializers.IntegerField(read_only=True) 3 title = serializers.CharField(max_length=32) 4 CHOICES = ((1, "Linux"), (2, "Django"), (3, "Python")) 5 chapter = serializers.ChoiceField(choices=CHOICES, source="get_chapter_display", read_only=True) 6 w_chapter = serializers.IntegerField(write_only=True) 7 pub_time = serializers.DateField() 8 date_added = serializers.DateField(write_only=True) 9 # 新增了一个上架时间字段 10 # 省略一些字段。。都是在原基础代码上增加的 11 # 。。。。。。 12 13 # 对多个字段进行验证 要求上架日期不能早于出版日期 上架日期要大 14 def validate(self, attrs): 15 if attrs["pub_time"] > attrs["date_added"]: 16 raise serializers.ValidationError("上架日期不能早于出版日期") 17 return attrs
1 def my_validate(value): 2 if "敏感词汇" in value.lower: 3 raise serializers.ValidationError("包含敏感词汇,请重新提交") 4 return value 5 6 7 class BookSerializer(serializers.Serializer): 8 id = serializers.IntegerField(read_only=True) 9 title = serializers.CharField(max_length=32, validators=[my_validate]) 10 # 。。。。。。 11
ModelSerializer
现在我们已经清楚了Serializer的用法,会发现我们所有的序列化跟我们的模型都紧密相关~
那么,DRF也给我们提供了跟模型紧密相关的序列化器~~ModelSerializer~~
-- 它会根据模型自动生成一组字段
-- 它简单的默认实现了.update()以及.create()方法
定义一个ModelSerializer序列化器
1 class BookSerializer(serializers.ModelSerializer): 2 class Meta: 3 model = Book 4 fields = "__all__" 5 # fields = ["id", "title", "pub_time"] 6 # exclude = ["user"] 7 # 分别是所有字段 包含某些字段 排除某些字段
外键关系的序列化
注意:当序列化类MATE中定义了depth时,这个序列化类中引用字段(外键)则自动变为只读
1 class BookSerializer(serializers.ModelSerializer): 2 class Meta: 3 model = Book 4 fields = "__all__" 5 # fields = ["id", "title", "pub_time"] 6 # exclude = ["user"] 7 # 分别是所有字段 包含某些字段 排除某些字段 8 depth = 1 9 # depth 代表找嵌套关系的第几层
自定义字段
我们可以声明一些字段来覆盖默认字段,来进行自定制~
比如我们的选择字段,默认显示的是选择的key,我们要给用户展示的是value。
1 class BookSerializer(serializers.ModelSerializer): 2 chapter = serializers.CharField(source="get_chapter_display", read_only=True) 3 4 class Meta: 5 model = Book 6 fields = "__all__" 7 # fields = ["id", "title", "pub_time"] 8 # exclude = ["user"] 9 # 分别是所有字段 包含某些字段 排除某些字段 10 depth = 1
Meta中其它关键字参数
1 class BookSerializer(serializers.ModelSerializer): 2 chapter = serializers.CharField(source="get_chapter_display", read_only=True) 3 4 class Meta: 5 model = Book 6 fields = "__all__" 7 # fields = ["id", "title", "pub_time"] 8 # exclude = ["user"] 9 # 分别是所有字段 包含某些字段 排除某些字段 10 depth = 1 11 read_only_fields = ["id"] 12 extra_kwargs = {"title": {"validators": [my_validate,]}}
post以及patch请求
由于depth会让我们外键变成只读,所以我们再定义一个序列化的类,其实只要去掉depth就可以了~~
1 class BookSerializer(serializers.ModelSerializer): 2 chapter = serializers.CharField(source="get_chapter_display", read_only=True) 3 4 class Meta: 5 model = Book 6 fields = "__all__" 7 # fields = ["id", "title", "pub_time"] 8 # exclude = ["user"] 9 # 分别是所有字段 包含某些字段 排除某些字段 10 read_only_fields = ["id"] 11 extra_kwargs = {"title": {"validators": [my_validate,]}}
SerializerMethodField
外键关联的对象有很多字段我们是用不到的~都传给前端会有数据冗余~就需要我们自己去定制序列化外键对象的哪些字段~~
1 class BookSerializer(serializers.ModelSerializer): 2 chapter = serializers.CharField(source="get_chapter_display", read_only=True) 3 user = serializers.SerializerMethodField() 4 publisher = serializers.SerializerMethodField() 5 6 def get_user(self, obj): 7 # obj是当前序列化的book对象 8 users_query_set = obj.user.all() 9 return [{"id": user_obj.pk, "name": user_obj.name} for user_obj in users_query_set] 10 11 def get_publisher(self, obj): 12 publisher_obj = obj.publisher 13 return {"id": publisher_obj.pk, "title": publisher_obj.title} 14 15 class Meta: 16 model = Book 17 fields = "__all__" 18 # fields = ["id", "title", "pub_time"] 19 # exclude = ["user"] 20 # 分别是所有字段 包含某些字段 排除某些字段 21 read_only_fields = ["id"] 22 extra_kwargs = {"title": {"validators": [my_validate,]}}
用ModelSerializer改进上面Serializer的完整版
1 class BookSerializer(serializers.ModelSerializer): 2 dis_chapter = serializers.SerializerMethodField(read_only=True) 3 users = serializers.SerializerMethodField(read_only=True) 4 publishers = serializers.SerializerMethodField(read_only=True) 5 6 def get_users(self, obj): 7 # obj是当前序列化的book对象 8 users_query_set = obj.user.all() 9 return [{"id": user_obj.pk, "name": user_obj.name} for user_obj in users_query_set] 10 11 def get_publishers(self, obj): 12 publisher_obj = obj.publisher 13 return {"id": publisher_obj.pk, "title": publisher_obj.title} 14 15 def get_dis_chapter(self, obj): 16 return obj.get_chapter_display() 17 18 class Meta: 19 model = Book 20 # fields = "__all__" 21 # 字段是有序的 22 fields = ["id", "title","dis_chapter", "pub_time", "publishers", "users","chapter", "user", "publisher"] 23 # exclude = ["user"] 24 # 分别是所有字段 包含某些字段 排除某些字段 25 read_only_fields = ["id", "dis_chapter", "users", "publishers"] 26 extra_kwargs = {"title": {"validators": [my_validate,]}, "user": {"write_only": True}, "publisher": {"write_only": True}, 27 "chapter": {"write_only": True}}

浙公网安备 33010602011771号