【Django】drf serializer序列化

model中字段属于外键的情况,序列化想展示外键表中的某个字段

A表的model结构如下

class OTAProductModel(BaseMysqlModel):
    id = models.AutoField(primary_key=True, verbose_name="id")
    ota = models.ForeignKey('OTAModel', on_delete=models.CASCADE, related_name='ota_product', verbose_name="平台")
    name = models.CharField(max_length=64, verbose_name="产品名称")
    code = models.CharField(max_length=64, unique=True, verbose_name="产品编码")
    history = HistoricalRecords(excluded_fields=['create_time', 'update_time'])

    class Meta:
        verbose_name = '平台产品管理'
        verbose_name_plural = verbose_name

    def __str__(self):
        return '%s(%s)' % (self.name, self.ota)

关联的外键表 结构如下


class OTAModel(BaseMysqlModel):
    id = models.AutoField(primary_key=True, verbose_name="id")
    name = models.CharField(max_length=64, verbose_name="平台名称")
    code = models.CharField(max_length=64, verbose_name="平台编号")
    history = HistoricalRecords(excluded_fields=['create_time', 'update_time'])

    class Meta:
        verbose_name = '平台管理'
        verbose_name_plural = verbose_name

    def __str__(self):
        return '%s(%s)' % (self.name, self.code)

想要的效果,在序列化A表时,展示出外键表B中的name,code字段,可以这样操作
在A表的序列话文件中

class OTAProductSerializer(serializers.ModelSerializer):
    create_time = serializers.DateTimeField(format='%Y-%m-%d %H:%M:%S', read_only=True)
    update_time = serializers.DateTimeField(format='%Y-%m-%d %H:%M:%S', read_only=True)
    ota_name = serializers.CharField(source='ota.name', read_only=True)
    ota_code = serializers.CharField(source='ota.code', read_only=True)
    message = OtaProductPromptMessageSerializer(many=True)

    class Meta:
        model = OTAProductModel

        fields = '__all__'

ota_name = serializers.CharField(source=‘ota.name’, read_only=True),需要注意两个地方,其中ota.name的ota是A表的model字段名称,name是外键表B中的字段名称

某些序列化字段需要特殊处理,可以自定义方法处理

from rest_framework import serializers

from project.models.project import ProjectModel


class ProjectSerializer(serializers.ModelSerializer):
    principal_username = serializers.CharField(source='project_principal.cn', read_only=True)
    progress_word = serializers.SerializerMethodField()

    class Meta:
        model = ProjectModel

        exclude = ['create_time', 'update_time']

    def get_progress_word(self, instance):
        progress = instance.progress
        word = ''
        if 0 <= progress and progress <= 5:
            word = '购前论证'
        elif 5 < progress and progress <= 15:
            word = '立项请示'
        elif 15 < progress and progress <= 30:
            word = '事前申请'
        elif 30 < progress and progress < 40:
            word = '采购计划'
        elif 40 < progress and progress <= 55:
            word = '招标采购'
        elif 55 < progress and progress <= 60:
            word = '中标'
        elif 60 < progress and progress <= 65:
            word = '合同签订'
        elif 65 < progress and progress <= 70:
            word = '合同备案'
        elif 70 < progress and progress <= 75:
            word = '项目实施'
        elif 75 < progress and progress <= 80:
            word = '验收'
        elif 80 < progress and progress <= 90:
            word = '监管验收'
        elif 90 < progress and progress <= 95:
            word = '固定资产报增'
        elif 95 < progress:
            word = '报账'
        return word

在这里插入图片描述
在这里插入图片描述

需要注意的地方,如上图

model 字段有choices,如何序列化

model表

class PaymentModel(BaseMongoModel):
    """
    支付列表
    """
    SALETYPE = ((1, u'一次销售'),
                (2, u'二次销售'))
    sale_type = IntField(required=True, verbose_name='支付订单的订单类型', default=1, choices=SALETYPE)
    class Meta:
        verbose_name = '支付记录'
        verbose_name_plural = verbose_name

    def __str__(self):
        return '%s' % (self.serial_number,)

序列化sale_type字段,可以这么操作

class PaymentSerializer(ser.DocumentSerializer):
    sale_type_name = serializers.CharField(source='get_sale_type_display', read_only=True)

    class Meta:
        model = PaymentModel
        fields = '__all__'

需要注意:source的格式:get_model的字段名_display

model 多对多字段的序列化

model

class EntranceModel(BaseModel):
    name = models.CharField(max_length=128, verbose_name="入口名称")
    depts = models.ManyToManyField('user.DeptModel', related_name='entrances', verbose_name='部门')
    longitude = models.FloatField(null=True, blank=True, verbose_name='经度')
    latitude = models.FloatField(null=True, blank=True, verbose_name='纬度')

    class Meta:
        verbose_name = '部门入口配置表'
        verbose_name_plural = verbose_name
        app_label = 'entrance'

Serializer

import logging

from rest_framework import serializers

from entrance.models import EntranceModel
from user.models import DeptModel

logger = logging.getLogger('system')
class EntranceSerializer(serializers.ModelSerializer):
    depts = serializers.PrimaryKeyRelatedField(many=True, allow_empty=True, allow_null=True, required=False,
                                               queryset=DeptModel.objects.all())
    depts_name = serializers.SerializerMethodField()

    class Meta:
        model = EntranceModel
        fields = '__all__'

    def get_depts_name(self, instance):
        names = instance.depts.all().values_list('name', flat=True)
        return ','.join(names) if names else None
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
DRFDjango Rest Framework)是一种用于构建Web API的强大框架,它提供了丰富的功能和工具来简化API的开发过程。在使用DRF时,我们经常需要定义视图和序列化器以及它们之间的关系来处理不同的请求和操作。 DRF的请求处理基于视图的操作方法,它们分别对应于HTTP方法,例如GET,POST,PUT,PATCH和DELETE。这些操作方法在视图中以函数的形式定义,并且通过装饰器或路由来映射到相应的URL。 序列化器是DRF的一个核心概念,它提供了一种简单而灵活的方式,将数据对象转换为可以被序列化和反序列化的格式,通常是JSON或XML。序列化器可以定义在视图中,也可以单独定义为一个类,然后在视图中使用。 在DRF中,我们可以通过定义不同的序列化器类来处理不同的操作。序列化器类可以继承自DRF提供的Serializer类或ModelSerializer类,后者自动为我们根据模型生成序列化器字段。 使用序列化器,我们可以在视图中方便地对请求进行验证、数据转换和结果序列化等操作。在视图的操作方法中,我们可以通过调用序列化器的不同方法来进行这些操作。 例如,我们可以使用序列化器的`is_valid()`方法来验证请求的数据是否有效。我们还可以使用`save()`方法来保存数据,或者使用`data`属性来获取序列化后的数据。此外,我们还可以通过调用`serializer_class()`方法来获取当前视图使用的序列化器类。 总之,DRF的action序列化提供了一种方便而强大的方式来处理API的请求和操作。通过定义视图和序列化器,并将它们联系起来,我们可以在开发API时更加高效和灵活。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值