Serialzers 序列化组件
前言
- 当作先后端分离的项目时,JSON是一个轻量级的数据交互格式。全部咱们给前端数据的时候都要转成json格式,那就须要对咱们从数据库拿到数据进行序列化
Django的序列化方法
- 使用django,json转数据进行传输,(了解便可)
class BooksView(View): def get(self, request): # 获取数据库中的queryset数据 book_list = Book.objects.values("id", "title", "chapter", "pub_time", "publisher") # 强转成列表类型 book_list = list(book_list) # 若是咱们须要取外键关联的字段信息 须要循环获取外键 再去数据库查而后拼接成咱们想要的 ret = [] for book in book_list: pub_dict = {} pub_obj = Publish.objects.filter(pk=book["publisher"]).first() pub_dict["id"] = pub_obj.pk pub_dict["title"] = pub_obj.title book["publisher"] = pub_dict ret.append(book) ret = json.dumps(book_list, ensure_ascii=False, cls=MyJson) return HttpResponse(ret) # json.JSONEncoder.default() # 解决json不能序列化时间字段的问题 class MyJson(json.JSONEncoder): def default(self, field): if isinstance(field, datetime.datetime): return field.strftime('%Y-%m-%d %H:%M:%S') elif isinstance(field, datetime.date): return field.strftime('%Y-%m-%d') else: return json.JSONEncoder.default(self, field)
- 因为太过于麻烦,全部咱们可使用DRF的序列化
字段和选项
- 注意serializer不是只能为数据库模型类定义,也能够为非数据库模型定义。serializer是独立于数据库以外的存在
1.字段和选项
<table style="border-spacing: 0px; font-size: 16px; color: #333333; font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif; letter-spacing: .2px; background-color: #ffffff;"> <thead> <tr style="border-top-color: #cccccc;"><th>字段</th><th>字段构造方式</th></tr> </thead> <tbody> <tr style="border-top-color: #cccccc;"> <td><span style="font-weight: bold;">BooleanField</span></td> <td>BooleanField()</td> </tr> <tr style="background-color: #f8f8f8; border-top-color: #cccccc;"> <td><span style="font-weight: bold;">NullBooleanField</span></td> <td>NullBooleanField()</td> </tr> <tr style="border-top-color: #cccccc;"> <td><span style="font-weight: bold;">CharField</span></td> <td>CharField(max_length=None, min_length=None, allow_blank=False, trim_whitespace=True)</td> </tr> <tr style="background-color: #f8f8f8; border-top-color: #cccccc;"> <td><span style="font-weight: bold;">EmailField</span></td> <td>EmailField(max_length=None, min_length=None, allow_blank=False)</td> </tr> <tr style="border-top-color: #cccccc;"> <td><span style="font-weight: bold;">RegexField</span></td> <td>RegexField(regex, max_length=None, min_length=None, allow_blank=False)</td> </tr> <tr style="background-color: #f8f8f8; border-top-color: #cccccc;"> <td><span style="font-weight: bold;">SlugField</span></td> <td>SlugField(max<span>length=50, min_length=None, allow_blank=False) <br />正则字段,验证正则模式 [a-zA-Z0-9</span>-]+</td> </tr> <tr style="border-top-color: #cccccc;"> <td><span style="font-weight: bold;">URLField</span></td> <td>URLField(max_length=200, min_length=None, allow_blank=False)</td> </tr> <tr style="background-color: #f8f8f8; border-top-color: #cccccc;"> <td><span style="font-weight: bold;">UUIDField</span></td> <td>UUIDField(format='hex_verbose') <br />format: <br />1) <code style="font-family: Consolas, 'Liberation Mono', Menlo, Courier, monospace; font-size: .85em; border: none; color: inherit; background-color: #f7f7f7;">'hex_verbose'</code> 如<code style="font-family: Consolas, 'Liberation Mono', Menlo, Courier, monospace; font-size: .85em; border: none; color: inherit; background-color: #f7f7f7;">"5ce0e9a5-5ffa-654b-cee0-1238041fb31a"</code> <br />2) <code style="font-family: Consolas, 'Liberation Mono', Menlo, Courier, monospace; font-size: .85em; border: none; color: inherit; background-color: #f7f7f7;">'hex'</code> 如 <code style="font-family: Consolas, 'Liberation Mono', Menlo, Courier, monospace; font-size: .85em; border: none; color: inherit; background-color: #f7f7f7;">"5ce0e9a55ffa654bcee01238041fb31a"</code> <br />3)<code style="font-family: Consolas, 'Liberation Mono', Menlo, Courier, monospace; font-size: .85em; border: none; color: inherit; background-color: #f7f7f7;">'int'</code> - 如: <code style="font-family: Consolas, 'Liberation Mono', Menlo, Courier, monospace; font-size: .85em; border: none; color: inherit; background-color: #f7f7f7;">"123456789012312313134124512351145145114"</code> <br />4)<code style="font-family: Consolas, 'Liberation Mono', Menlo, Courier, monospace; font-size: .85em; border: none; color: inherit; background-color: #f7f7f7;">'urn'</code> 如: <code style="font-family: Consolas, 'Liberation Mono', Menlo, Courier, monospace; font-size: .85em; border: none; color: inherit; background-color: #f7f7f7;">"urn:uuid:5ce0e9a5-5ffa-654b-cee0-1238041fb31a"</code></td> </tr> <tr style="border-top-color: #cccccc;"> <td><span style="font-weight: bold;">IPAddressField</span></td> <td>IPAddressField(protocol='both', unpack_ipv4=False, **options)</td> </tr> <tr style="background-color: #f8f8f8; border-top-color: #cccccc;"> <td><span style="font-weight: bold;">IntegerField</span></td> <td>IntegerField(max_value=None, min_value=None)</td> </tr> <tr style="border-top-color: #cccccc;"> <td><span style="font-weight: bold;">FloatField</span></td> <td>FloatField(max_value=None, min_value=None)</td> </tr> <tr style="background-color: #f8f8f8; border-top-color: #cccccc;"> <td><span style="font-weight: bold;">DecimalField</span></td> <td>DecimalField(max_digits, decimal_places, coerce_to_string=None, max_value=None, min_value=None)<br />max_digits: 最多位数<br />decimal_palces: 小数点位置</td> </tr> <tr style="border-top-color: #cccccc;"> <td><span style="font-weight: bold;">DateTimeField</span></td> <td>DateTimeField(format=api_settings.DATETIME_FORMAT, input_formats=None)</td> </tr> <tr style="background-color: #f8f8f8; border-top-color: #cccccc;"> <td><span style="font-weight: bold;">DateField</span></td> <td>DateField(format=api_settings.DATE_FORMAT, input_formats=None)</td> </tr> <tr style="border-top-color: #cccccc;"> <td><span style="font-weight: bold;">TimeField</span></td> <td>TimeField(format=api_settings.TIME_FORMAT, input_formats=None)</td> </tr> <tr style="background-color: #f8f8f8; border-top-color: #cccccc;"> <td><span style="font-weight: bold;">DurationField</span></td> <td>DurationField()</td> </tr> <tr style="border-top-color: #cccccc;"> <td><span style="font-weight: bold;">ChoiceField</span></td> <td>ChoiceField(choices)<br />choices与Django的用法相同</td> </tr> <tr style="background-color: #f8f8f8; border-top-color: #cccccc;"> <td><span style="font-weight: bold;">MultipleChoiceField</span></td> <td>MultipleChoiceField(choices)</td> </tr> <tr style="border-top-color: #cccccc;"> <td><span style="font-weight: bold;">FileField</span></td> <td>FileField(max_length=None, allow_empty_file=False, use_url=UPLOADED_FILES_USE_URL)</td> </tr> <tr style="background-color: #f8f8f8; border-top-color: #cccccc;"> <td><span style="font-weight: bold;">ImageField</span></td> <td>ImageField(max_length=None, allow_empty_file=False, use_url=UPLOADED_FILES_USE_URL)</td> </tr> <tr style="border-top-color: #cccccc;"> <td><span style="font-weight: bold;">ListField</span></td> <td>ListField(child=, min_length=None, max_length=None)</td> </tr> <tr style="background-color: #f8f8f8; border-top-color: #cccccc;"> <td><span style="font-weight: bold;">DictField</span></td> <td>DictField(child=)</td> </tr> </tbody> </table>html
2.选项参数
<table style="border-spacing:0px;font-size:16px;color:rgb(51,51,51);font-family:'Helvetica Neue', Helvetica, Arial, sans-serif;letter-spacing:.2px;background-color:rgb(255,255,255);"><thead><tr style="border-top-color:rgb(204,204,204);"><th>参数名称</th><th>做用</th></tr></thead><tbody><tr style="border-top-color:rgb(204,204,204);"><td><span style="font-weight:700;">max_length</span></td><td>最大长度</td></tr><tr style="background-color:rgb(248,248,248);border-top-color:rgb(204,204,204);"><td><span style="font-weight:700;">min_lenght</span></td><td>最小长度</td></tr><tr style="border-top-color:rgb(204,204,204);"><td><span style="font-weight:700;">allow_blank</span></td><td>是否容许为空</td></tr><tr style="background-color:rgb(248,248,248);border-top-color:rgb(204,204,204);"><td><span style="font-weight:700;">trim_whitespace</span></td><td>是否截断空白字符</td></tr><tr style="border-top-color:rgb(204,204,204);"><td><span style="font-weight:700;">max_value</span></td><td>最小值</td></tr><tr style="background-color:rgb(248,248,248);border-top-color:rgb(204,204,204);"><td><span style="font-weight:700;">min_value</span></td><td>最大值</td></tr></tbody></table>前端
3.通用参数
<table style="border-spacing:0px;font-size:16px;color:rgb(51,51,51);font-family:'Helvetica Neue', Helvetica, Arial, sans-serif;letter-spacing:.2px;background-color:rgb(255,255,255);"><thead><tr style="border-top-color:rgb(204,204,204);"><th>参数名称</th><th>说明</th></tr></thead><tbody><tr style="border-top-color:rgb(204,204,204);"><td><span style="font-weight:700;">read_only</span></td><td>代表该字段仅用于序列化输出,默认False</td></tr><tr style="background-color:rgb(248,248,248);border-top-color:rgb(204,204,204);"><td><span style="font-weight:700;">write_only</span></td><td>代表该字段仅用于反序列化输入,默认False</td></tr><tr style="border-top-color:rgb(204,204,204);"><td><span style="font-weight:700;">required</span></td><td>代表该字段在反序列化时必须输入,默认True</td></tr><tr style="background-color:rgb(248,248,248);border-top-color:rgb(204,204,204);"><td><span style="font-weight:700;">default</span></td><td>反序列化时使用的默认值</td></tr><tr style="border-top-color:rgb(204,204,204);"><td><span style="font-weight:700;">allow_null</span></td><td>代表该字段是否容许传入None,默认False</td></tr><tr style="background-color:rgb(248,248,248);border-top-color:rgb(204,204,204);"><td><span style="font-weight:700;">validators</span></td><td>该字段使用的验证器</td></tr><tr style="border-top-color:rgb(204,204,204);"><td><span style="font-weight:700;">error_messages</span></td><td>包含错误编号与错误信息的字典</td></tr><tr style="background-color:rgb(248,248,248);border-top-color:rgb(204,204,204);"><td><span style="font-weight:700;">label</span></td><td>用于HTML展现API页面时,显示的字段名称</td></tr><tr style="border-top-color:rgb(204,204,204);"><td><span style="font-weight:700;">help_text</span></td><td>用于HTML展现API页面时,显示的字段帮助提示信息</td></tr></tbody></table>python
DRF序列化的方法
1. 自定义序列化
-
须要自定义全部字段,以及方法create,updata方法git
-
声明一个序列化类数据库
# 根据model.py中的字段对应写要显示的字段 class BookSerializer(serializers.Serializer): id = serializers.IntegerField() title = serializers.CharField(max_length=32) CHOICES = ((1, "Linux"), (2, "Django"), (3, "Python")) chapter = serializers.ChoiceField(choices=CHOICES, source="get_chapter_display") pub_time = serializers.DateField()
-
将数据库中的数据传入序列化对象中django
from rest_framework.views import APIView from rest_framework.response import Response class BookView(APIView): def get(self, request): # 获取数据库中的数据 book_list = Book.objects.all() # 实例化序列化对象, many= True 表示数据有多个,若是是一个则不须要 ret = BookSerializer(book_list, many=True) # 给前端的数据 return Response(ret.data)
2. 外键关系的序列化
from rest_framework import serializers from .models import Book class PublisherSerializer(serializers.Serializer): # read_only=True 表示的是给前端时显示,反序列时,不须要此字段 id = serializers.IntegerField(read_only=True) title = serializers.CharField(max_length=32) class UserSerializer(serializers.Serializer): id = serializers.IntegerField(read_only=True) name = serializers.CharField(max_length=32) age = serializers.IntegerField() class BookSerializer(serializers.Serializer): id = serializers.IntegerField(read_only=True) title = serializers.CharField(max_length=32) CHOICES = ((1, "Linux"), (2, "Django"), (3, "Python")) chapter = serializers.ChoiceField(choices=CHOICES, source="get_chapter_display", read_only=True) pub_time = serializers.DateField() publisher = PublisherSerializer(read_only=True) # many= True 表示有多条数据 user = UserSerializer(many=True, read_only=True)
3. 反序列化
-
当前端给咱们发post请求的时候,前端给后端传数据的,咱们须要对数据进行一些校验而后保存到数据库,或者对数据库中的数据进行更改,DRF的serializer也提供了方法json
-
Serializer提供了is_valid()和save方法后端
-
url.pyapi
urlpatterns = [ url(r'^list$', BooksView.as_view()), ]
-
声明一个序列化类app
# serializers.py 文件 class BookSerializer(serializers.Serializer): # read_only = True 表示前端不须要传该字段的数据,其余的都须要传 id = serializers.IntegerField(read_only=True) title = serializers.CharField(max_length=32) CHOICES = ((1, "Linux"), (2, "Django"), (3, "Python")) chapter = serializers.ChoiceField(choices=CHOICES, source="get_chapter_display", read_only=True) # write_only = True 表示前端须要传这个字段的数据, w_chapter = serializers.IntegerField(write_only=True) pub_time = serializers.DateField() publisher = PublisherSerializer(read_only=True) user = UserSerializer(many=True, read_only=True) users = serializers.ListField(write_only=True) publisher_id = serializers.IntegerField(write_only=True) # post请求会执行该方法 def create(self, validated_data): 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"]) book.user.add(*validated_data["users"]) # 须要返回对象 # 若是数据简单, # book = Book.objects.create(**validated_data) return book
- 注意: 由于存在后端给前端显示的数据和前端给后端存储的数据有区别,全部要灵活运用read_only= True 和write_only=True ,俩个属性,
-
反序列化
class BookView(APIView): def get(self, request): book_list = Book.objects.all() ret = BookSerializer(book_list, many=True) return Response(ret.data) # 会执行序列化中的create方法 def post(self, request): # book_obj = request.data print(request.data) # 将前端传来的数据传入序列化对象中 serializer = BookSerializer(data=request.data) # 判断数据 if serializer.is_valid(): # 执行后端的create方法 serializer.save() return Response(serializer.validated_data) # 不然将错误信息发回给前端 return Response(serializer.errors)
-
对数据进行编辑
url.py
urlpatterns = [ url(r'book/(?P<pk>\d+)/', BookView.as_view(), name='book'), ]
声明一个序列化类
class BookSerializer(serializers.Serializer): id = serializers.IntegerField(read_only=True) title = serializers.CharField(max_length=32) CHOICES = ((1, "Linux"), (2, "Django"), (3, "Python")) chapter = serializers.ChoiceField(choices=CHOICES, source="get_chapter_display", read_only=True) w_chapter = serializers.IntegerField(write_only=True) pub_time = serializers.DateField() publisher = PublisherSerializer(read_only=True) user = UserSerializer(many=True, read_only=True) users = serializers.ListField(write_only=True) publisher_id = serializers.IntegerField(write_only=True) def create(self, validated_data): 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"]) book.user.add(*validated_data["users"]) return book # put或者patch或执行该方法 def update(self, instance, validated_data): instance.title = validated_data.get("title", instance.title) instance.chapter = validated_data.get("w_chapter", instance.chapter) instance.pub_time = validated_data.get("pub_time", instance.pub_time) instance.publisher_id = validated_data.get("publisher_id", instance.publisher_id) if validated_data.get("users"): instance.user.set(validated_data.get("users")) instance.save() return instance
view.py
class BookView(APIView): def patch(self, request): print(request.data) book_id = request.data["id"] book_info = request.data["book_info"] book_obj = Book.objects.filter(pk=book_id).first() # partial= True 对部分数据进行修改 serializer = BookSerializer(book_obj, data=book_info, partial=True) if serializer.is_valid(): serializer.save() return Response(serializer.data) else: return Response(serializer.errors) # 这俩种方法是同样的只写其中一种 def put(self, request, id): book_obj = Book.objects.filter(id=id).first() ser_obj = BookSerializer(instance=book_obj, data=request.data, partial=True) if ser_obj.is_valid(): ser_obj.save() return Response(ser_obj.data) return Response(ser_obj.errors)
3.1 view四个方法的运用
-- get 请求 def get(self, request): queryset = Book.objects.all() ser_obj = BookSerializer(queryset, many=True) return Response(ser_obj.data) -- post请求 def post(self, request): data = request.data ser_obj = BookSerializer(data=request.data) if ser_obj.is_valid(): ser_obj.save() # ser_obj.validated_data return Response(ser_obj.data) else: return Response(ser_obj.errors) -- put/patch请求 def put(self, request, id): data = request.data book_obj = Book.objects.filter(id=id).first() ser_obj = BookSerializer(instance=book_obj, data=request.data, partial=True) # partial表示局部更新 if ser_obj.is_valid(): ser_obj.save() # ser_obj.validated_data return Response(ser_obj.data) else: return Response(ser_obj.errors)
4.验证
-
优先级从高到低: 自定义>单独验证>多个字段验证
-
若是咱们须要对某些字段进行自定义验证,DRF也给咱们提供了钩子方法
-
一 对单个数据进行验证, 格式必须是validate_须要验证的字段名(self, value):
class BookSerializer(serializers.Serializer): id = serializers.IntegerField(read_only=True) title = serializers.CharField(max_length=32) # 省略了一些字段 跟上面代码里同样的 # 。。。。。 # 格式必须是validate_须要验证的字段名(self, value) def validate_title(self, value): if "python" not in value.lower(): raise serializers.ValidationError("标题必须含有Python") return value
-
对多个数据进行验证
class BookSerializer(serializers.Serializer): id = serializers.IntegerField(read_only=True) title = serializers.CharField(max_length=32) CHOICES = ((1, "Linux"), (2, "Django"), (3, "Python")) chapter = serializers.ChoiceField(choices=CHOICES, source="get_chapter_display", read_only=True) w_chapter = serializers.IntegerField(write_only=True) pub_time = serializers.DateField() date_added = serializers.DateField(write_only=True) # 新增了一个上架时间字段 # 省略一些字段。。都是在原基础代码上增长的 # 。。。。。。 # 对多个字段进行验证 要求上架日期不能早于出版日期 上架日期要大 def validate(self, attrs): if attrs["pub_time"] > attrs["date_added"]: raise serializers.ValidationError("上架日期不能早于出版日期") return attrs
-
定义一个验证器
def my_validate(value): if "敏感词汇" in value.lower: raise serializers.ValidationError("包含敏感词汇,请从新提交") return value class BookSerializer(serializers.Serializer): id = serializers.IntegerField(read_only=True) title = serializers.CharField(max_length=32, validators=[my_validate]) # 。。。。。。
ModelSerializer
-
如今咱们已经清楚了Serializer的用法,会发现咱们全部的序列化跟咱们的模型都紧密相关~
那么,DRF也给咱们提供了跟模型紧密相关的序列化器ModelSerializer
-- 它会根据模型自动生成一组字段
-- 它简单的默认实现了.update()以及.create()方法
1.定义一个ModelSerializer序列化器
class BookSerializer(serializers.ModelSerializer): class Meta: model = Book fields = "__all__" # fields = ["id", "title", "pub_time"] # exclude = ["user"] # 分别是全部字段 包含某些字段 排除某些字段
2.外键关联的序列化
自动序列化连表操做,可使用depth来进行快捷的跨表操做,官方建议是0~10层,可是最好用到3或者4层就能够了
注意:当序列化类MATE中定义了depth时,这个序列化类中引用字段(外键)则自动变为只读
class BookSerializer(serializers.ModelSerializer): class Meta: model = Book fields = "__all__" # fields = ["id", "title", "pub_time"] # exclude = ["user"] # 分别是全部字段 包含某些字段 排除某些字段 depth = 1 # depth 表明找嵌套关系的第几层
3.自定义字段对覆盖默认字段
咱们能够声明一些字段来覆盖默认字段,来进行自定制~
好比咱们的选择字段,默认显示的是选择的key,咱们要给用户展现的是value。
class BookSerializer(serializers.ModelSerializer): chapter = serializers.CharField(source="get_chapter_display", read_only=True) class Meta: model = Book fields = "__all__" # fields = ["id", "title", "pub_time"] # exclude = ["user"] # 分别是全部字段 包含某些字段 排除某些字段 depth = 1
4.meta中的其它关键字参数
class BookSerializer(serializers.ModelSerializer): chapter = serializers.CharField(source="get_chapter_display", read_only=True) class Meta: model = Book fields = "__all__" # fields = ["id", "title", "pub_time"] # exclude = ["user"] # 分别是全部字段 包含某些字段 排除某些字段 depth = 1 # read_only_fields 表示只读 read_only_fields = ["id"] extra_kwargs = {"title": {"validators": [my_validate,]}}
-
read_only_fields 表示id为只读,
-
extra_kwargs
选项在字段上指定任意附加关键字参数。与read_only_fields
的状况同样,这意味着你不须要在序列化类中显式声明该字段。 -
该选项是一个字典,将字段名称映射到关键字参数字典。例如:
class CreateUserSerializer(serializers.ModelSerializer): class Meta: model = User fields = ('email', 'username', 'password') extra_kwargs = {'password': {'write_only': True}}
5 post以及patch/put请求
- 因为depth会让咱们外键变成只读,因此咱们再定义一个序列化的类,其实只要去掉depth就能够了~~
class BookSerializer(serializers.ModelSerializer): chapter = serializers.CharField(source="get_chapter_display", read_only=True) class Meta: model = Book fields = "__all__" # fields = ["id", "title", "pub_time"] # exclude = ["user"] # 分别是全部字段 包含某些字段 排除某些字段 read_only_fields = ["id"] extra_kwargs = {"title": {"validators": [my_validate,]}}
6 SerializerMethodField
- 针对于多对多表数据的显示,可使用SerializerMethodField,来自定义显示
外键关联的对象有不少字段咱们是用不到的~都传给前端会有数据冗余~就须要咱们本身去定制序列化外键对象的哪些字段~~
class BookSerializer(serializers.ModelSerializer): chapter = serializers.CharField(source="get_chapter_display", read_only=True) user = serializers.SerializerMethodField() publisher = serializers.SerializerMethodField() def get_user(self, obj): # obj是当前序列化的book对象 users_query_set = obj.user.all() return [{"id": user_obj.pk, "name": user_obj.name} for user_obj in users_query_set] def get_publisher(self, obj): publisher_obj = obj.publisher return {"id": publisher_obj.pk, "title": publisher_obj.title} class Meta: model = Book fields = "__all__" # fields = ["id", "title", "pub_time"] # exclude = ["user"] # 分别是全部字段 包含某些字段 排除某些字段 read_only_fields = ["id"] extra_kwargs = {"title": {"validators": [my_validate,]}}
7. 用ModelSerializer改进上面Serializer的完整版
class BookSerializer(serializers.ModelSerializer): dis_chapter = serializers.SerializerMethodField(read_only=True) users = serializers.SerializerMethodField(read_only=True) publishers = serializers.SerializerMethodField(read_only=True) def get_users(self, obj): # obj是当前序列化的book对象 users_query_set = obj.user.all() return [{"id": user_obj.pk, "name": user_obj.name} for user_obj in users_query_set] def get_publishers(self, obj): publisher_obj = obj.publisher return {"id": publisher_obj.pk, "title": publisher_obj.title} def get_dis_chapter(self, obj): return obj.get_chapter_display() class Meta: model = Book # fields = "__all__" # 字段是有序的 fields = ["id", "title","dis_chapter", "pub_time", "publishers", "users","chapter", "user", "publisher"] # exclude = ["user"] # 分别是全部字段 包含某些字段 排除某些字段 read_only_fields = ["id", "dis_chapter", "users", "publishers"] extra_kwargs = {"title": {"validators": [my_validate,]}, "user": {"write_only": True}, "publisher": {"write_only": True}, "chapter": {"write_only": True}}
返回URL的链接
详情见网址:https://www.cnblogs.com/wupeiqi/articles/7805382.html中的序列化中的url