django ModelSerializer自定义显示字段


前言

最近在复习django的时候,发现了一个有趣的问题,解决了之后特意记录下来,以供以后参考。

一、问题

相信大家使用django的时候,被其DRF的强大功能所折服,因为它能通过简单的代码就能帮助我们实现增删改查等最简单的操作。我的demo代码如下:
模型类代码,models.py

from django.db import models


class StudentInfo(models.Model):
    name = models.CharField(max_length=10, verbose_name='姓名')
    sex = models.CharField(max_length=1, verbose_name='性别')
    from_class = models.ForeignKey(ClassInfo, on_delete=models.CASCADE)

序列化器代码,serializers.py

from rest_framework import serializers
from demo import models


class StudentInfoSerializer(serializers.ModelSerializer):
    class Meta:
        model = models.StudentInfo
        fields ="__all__"

视图代码,views.py

from rest_framework.viewsets import ModelViewSet
from .models import StudentInfo
from .serializers import StudentInfoSerializer


class DemoView(ModelViewSet):
    queryset = StudentInfo.objects.all()
    serializer_class = StudentInfoSerializer

路由代码,urls.py

from rest_framework.routers import SimpleRouter, DefaultRouter
from .views import DemoView
urlpatterns = []

# demo_route = SimpleRouter()
demo_route = DefaultRouter()

demo_route.register("demo", DemoView, basename="demo")

urlpatterns += demo_route.urls

最后在项目的主路由里面添加路径即可(需要在settings.py里面注册app):

from django.contrib import admin
from django.urls import path, include

urlpatterns = [
    path("admin/", admin.site.urls),
    path("user/", include("user.urls")),
    path("", include("demo.urls"))
]

效果:
在这里插入图片描述
但是,这样会出现一个问题,就是无法控制序列化结果显示的字段。例如,在同一个项目中,可能会出现多个场景,一个场景只需要用户的name和sex字段,一个场景只需要用户的name字段,一个场景则需要用户的全部字段,按照以上方法就需要设置三个序列化器了,显然不符合实际应用。那么有没有办法可以兼顾便捷和灵活呢?

二、解决

解决方法很简单,重写序列化器的__init__()方法即可:
serializers.py

# 动态修改fileds字段
class StudentInfoSerializer(serializers.ModelSerializer):
    """
         此处的`fields`字段是用来替换上面Serializer内部Meta类中指定的`fields`属性值
        """

    def __init__(self, *args, **kwargs):
        # 在super执行之前
        # 将传递的`fields`中的字段从kwargs取出并剔除,避免其传递给基类ModelSerializer
        # 注意此处`fields`中在默认`self.fields`属性中不存在的字段将无法被序列化 也就是`fields`中的字段应该是`self.fields`的子集
        fields = kwargs.pop('fields', None)

        super(StudentInfoSerializer, self).__init__(*args, **kwargs)

        if fields is not None:
            # 从默认`self.fields`属性中剔除非`fields`中指定的字段(两个集合相减,会提出多余的元素)
            allowed = set(fields)
            existing = set(self.fields.keys())
            for field_name in existing - allowed:
                self.fields.pop(field_name)

    class Meta:
        model = models.StudentInfo
        fields ="__all__"

以上代码的原理也很简单,首先获取传入的fields参数(即你想要的字段);然后使用序列化器原有的字段减去你想要的字段,就获取了多余的字段;最后,循环遍历多余的字段,将它们从原有的字段中一个一个剔除
为了更好的展示结果,我这里自定义了三个路径:
views.py

from django.shortcuts import render

# Create your views here.
from rest_framework.decorators import action
from rest_framework.response import Response
from rest_framework.viewsets import ModelViewSet
from .models import StudentInfo
from .serializers import StudentInfoSerializer


class DemoView(ModelViewSet):
    queryset = StudentInfo.objects.all()
    serializer_class = StudentInfoSerializer

    @action(url_path="all_fields", methods=["GET"], detail=False)
    def all_fields(self, request):
        user_data = StudentInfo.objects.all()
        serializer = StudentInfoSerializer(instance=user_data, many=True)

        return Response(serializer.data)

    @action(url_path="name_sex", methods=["GET"], detail=False)
    def name_sex(self, request):
        user_data = StudentInfo.objects.all()
        serializer = StudentInfoSerializer(instance=user_data, fields=("name", "sex"), many=True)

        return Response(serializer.data)

    @action(url_path="name_fromclass", methods=["GET"], detail=False)
    def name_fromclass(self, request):
        user_data = StudentInfo.objects.all()
        serializer = StudentInfoSerializer(instance=user_data, fields=("name", "from_class"), many=True)

        return Response(serializer.data)

效果如下:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
可以看出,根据传入的fields的不同,返回不同的结果。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值