django ModelSerializer自定义显示字段

2023-11-21 12:36

本文主要是介绍django ModelSerializer自定义显示字段,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 前言
  • 一、问题
  • 二、解决


前言

最近在复习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的不同,返回不同的结果。

这篇关于django ModelSerializer自定义显示字段的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/402394

相关文章

如何自定义一个log适配器starter

《如何自定义一个log适配器starter》:本文主要介绍如何自定义一个log适配器starter的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录需求Starter 项目目录结构pom.XML 配置LogInitializer实现MDCInterceptor

RedisTemplate默认序列化方式显示中文乱码的解决

《RedisTemplate默认序列化方式显示中文乱码的解决》本文主要介绍了SpringDataRedis默认使用JdkSerializationRedisSerializer导致数据乱码,文中通过示... 目录1. 问题原因2. 解决方案3. 配置类示例4. 配置说明5. 使用示例6. 验证存储结果7.

Druid连接池实现自定义数据库密码加解密功能

《Druid连接池实现自定义数据库密码加解密功能》在现代应用开发中,数据安全是至关重要的,本文将介绍如何在​​Druid​​连接池中实现自定义的数据库密码加解密功能,有需要的小伙伴可以参考一下... 目录1. 环境准备2. 密码加密算法的选择3. 自定义 ​​DruidDataSource​​ 的密码解密3

spring-gateway filters添加自定义过滤器实现流程分析(可插拔)

《spring-gatewayfilters添加自定义过滤器实现流程分析(可插拔)》:本文主要介绍spring-gatewayfilters添加自定义过滤器实现流程分析(可插拔),本文通过实例图... 目录需求背景需求拆解设计流程及作用域逻辑处理代码逻辑需求背景公司要求,通过公司网络代理访问的请求需要做请

idea中project的显示问题及解决

《idea中project的显示问题及解决》:本文主要介绍idea中project的显示问题及解决方案,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录idea中project的显示问题清除配置重China编程新生成配置总结idea中project的显示问题新建空的pr

对Django中时区的解读

《对Django中时区的解读》:本文主要介绍对Django中时区的解读方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录背景前端数据库中存储接口返回AI的解释问题:这样设置的作用答案获取当前时间(自动带时区)转换为北京时间显示总结背景设置时区为北京时间 TIM

Django之定时任务django-crontab的实现

《Django之定时任务django-crontab的实现》Django可以使用第三方库如django-crontab来实现定时任务的调度,本文主要介绍了Django之定时任务django-cront... 目录crontab安装django-crontab注册应用定时时间格式定时时间示例设置定时任务@符号

电脑显示mfc100u.dll丢失怎么办?系统报错mfc90u.dll丢失5种修复方案

《电脑显示mfc100u.dll丢失怎么办?系统报错mfc90u.dll丢失5种修复方案》最近有不少兄弟反映,电脑突然弹出“mfc100u.dll已加载,但找不到入口点”的错误提示,导致一些程序无法正... 在计算机使用过程中,我们经常会遇到一些错误提示,其中最常见的就是“找不到指定的模块”或“缺少某个DL

Spring Security自定义身份认证的实现方法

《SpringSecurity自定义身份认证的实现方法》:本文主要介绍SpringSecurity自定义身份认证的实现方法,下面对SpringSecurity的这三种自定义身份认证进行详细讲解,... 目录1.内存身份认证(1)创建配置类(2)验证内存身份认证2.JDBC身份认证(1)数据准备 (2)配置依

使用Sentinel自定义返回和实现区分来源方式

《使用Sentinel自定义返回和实现区分来源方式》:本文主要介绍使用Sentinel自定义返回和实现区分来源方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Sentinel自定义返回和实现区分来源1. 自定义错误返回2. 实现区分来源总结Sentinel自定