已解决
django ModelSerializer自定义显示字段
来自网友在路上 11158115提问 提问时间:2023-11-22 20:47:17阅读次数: 115
最佳答案 问答题库1158位专家为你答疑解惑
文章目录
- 前言
- 一、问题
- 二、解决
前言
最近在复习django的时候,发现了一个有趣的问题,解决了之后特意记录下来,以供以后参考。
一、问题
相信大家使用django的时候,被其DRF的强大功能所折服,因为它能通过简单的代码就能帮助我们实现增删改查等最简单的操作。我的demo代码如下:
模型类代码,models.py
from django.db import modelsclass 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 modelsclass StudentInfoSerializer(serializers.ModelSerializer):class Meta:model = models.StudentInfofields ="__all__"
视图代码,views.py
from rest_framework.viewsets import ModelViewSet
from .models import StudentInfo
from .serializers import StudentInfoSerializerclass 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, includeurlpatterns = [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.StudentInfofields ="__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 StudentInfoSerializerclass 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的不同,返回不同的结果。
查看全文
99%的人还看了
相似问题
- django ModelSerializer自定义显示字段
- 替换sql,某个字段特定容
- Java之反射获取和赋值字段
- java mybatisplus generator 修改字段类型
- 使用用户代理字段进行浏览器检测(判断页面运行环境)
- js数组操作——对象数组根据某个相同的字段分组
- spring boot加mybatis puls实现,在新增/修改时,对某些字段进行处理,使用的@TableField()
- 【IDEA 使用easyAPI、easyYapi、Apifox helper等插件时,导出接口文档缺少代码字段注释的相关内容、校验规则的解决方法】
- mysql取出组内按照某时间最新一条数据的其他字段
- 基于geotools24.0的创建自动增长主键id字段的方法
猜你感兴趣
版权申明
本文"django ModelSerializer自定义显示字段":http://eshow365.cn/6-41963-0.html 内容来自互联网,请自行判断内容的正确性。如有侵权请联系我们,立即删除!
- 上一篇: sso 四种授权模式
- 下一篇: lombok 引入