有关django序列化器的疑问

1、关于序列化器的create和update


instance = YourModel.objects.get(id=object_id)

serializer.update(instance, validated_data) 

代码:

def update(self, instance, validated_data):

    # 获取需要比较的字段的值

    field1_value = validated_data.get('field1')

    field2_value = validated_data.get('field2')

    # 根据条件判断是否需要更新

    if field1_value != field2_value:

        # 更新需要的字段

        instance.field1 = field1_value

        instance.field2 = field2_value

        # 执行其他更新操作

    # 保存更新后的实例

    instance.save()

    return instance

2、如何在视图里进行验证和保存(创建或更新)

def post(self, request):

    # 获取请求数据

    data = request.data

    # 检查是否已存在当天的数据

    existing_data = YourModel.objects.filter(date=date.today()).first()

    if existing_data:

        # 数据已存在,调用序列化器的 update 方法

        serializer = YourSerializer(existing_data, data=data)

    else:

        # 数据不存在,调用序列化器的 create 方法

        serializer = YourSerializer(data=data)

    if serializer.is_valid():

        serializer.save()

        return Response(serializer.data, status=status.HTTP_201_CREATED)



    return Response(serializer.errors, status=status.HTTP_400_BAD

如果使用 YourSerializer(data=data) 实例化方式,调用 serializer.save() 或 serializer.create(validated_data) 执行创建逻辑。

如果使用 YourSerializer(existing_data, data=data) 实例化方式,调用 serializer.save() 或 serializer.update(instance, validated_data) 执行更新逻辑。

其实serializer.save()内部逻辑会判断是否有instance来决定是走create还是update

这个键名就变成了validate_errors

序列化器数据校验顺序

1.对每个字段类型,参数进行校验 (is_valid())
2.调用局部钩子validate_字段名 进行校验(是一个写在序列化器内的方法,只能针对单个参数)
3.全局钩子validate进行校验(也在序列化器内)
4.使用字段选项中的validators执行外部函数的校验(可以写个validate.py)

depth就是嵌套序列化时的一个能获取外键的方法,下面这个是例子

https://blog.csdn.net/weixin_42134789/article/details/109064124

这个更详细

https://blog.csdn.net/qq_39147299/article/details/108691882

序列化器的工作流程

整个序列化器的工作步骤

1.初始化序列化器:创建序列化器实例,并将传入的数据作为参数进行初始化。
2.数据反序列化(To Internal Value):在保存对象之前,会执行序列化器的 to_internal_value() 方法。该方法将传入的外部数据进行反序列化,将其转换为内部表示形式。
3.字段验证(Field Validation):对每个字段进行验证,包括字段级别的验证器和字段类型验证。
4.对象验证(Object Validation):执行序列化器的 validate() 方法,验证整个对象的值或字段之间的关系。
5.保存对象(Save Object):执行序列化器的 save() 方法,将数据保存到数据库中。
6.数据序列化(To Representation):在保存对象后,执行序列化器的 to_representation() 方法,将保存后的对象转换为可序列化的表示形式。
这是序列化器在保存或更新数据时的完整流程:初始化序列化器 -> 数据反序列化 -> 字段验证 -> 对象验证 -> 保存对象 -> 数据序列化。每个步骤都有特定的目的和功能,确保数据的正确性和一致性。


一个例子:

从B,C模型里拿数据进行处理形成了a接口,想把a接口的部分数据写到A模型里,反序列化写入数据库的每一步需要怎么操作,看过to_representation,create,validate但就是串不起来,debug时也进不去

首先,让我们从serializer开始解释。Serializer是Django REST Framework(DRF)中用于序列化和反序列化数据的核心组件。它允许您将复杂的数据类型(例如,Django模型实例)转换为Python数据类型(例如,字典)或JSON格式,以便它们可以被传输和存储。

每个serializer类都包含了一些方法,其中一些是必须的,一些是可选的。下面是一些常用的和必须的方法:

  1. to_representation: 将模型实例转换为Python数据类型或JSON格式,以便它们可以被序列化。这个方法通常用于GET请求。

  2. to_internal_value: 将Python数据类型或JSON格式转换为模型实例。这个方法通常用于POST请求。

  3. validate: 在将数据保存到数据库之前,验证数据是否符合要求。这里可以进行一些自定义验证逻辑。

  4. create: 在验证通过后,将数据保存到数据库中。这个方法通常用于POST请求。

接下来,让我们回到你的问题。如果你想将a接口的部分数据写入A模型中,你可以创建一个名为A模型serializer的serializer类,并覆盖create方法。在create方法中,您可以将传入的数据反序列化为A模型的实例,并将其保存到数据库中。这个过程可以通过以下步骤完成:

  1. 创建A模型serializer类,继承自serializers.ModelSerializer
from rest_framework import serializers
from .models import A

class ASerializer(serializers.ModelSerializer):
    class Meta:
        model = A
        fields = '__all__'
  1. 在视图中使用ASerializer类来反序列化数据,并将其保存到A模型中。
from rest_framework import views, status
from rest_framework.response import Response
from .serializers import ASerializer
from .models import A

class MyAPIView(views.APIView):
    def post(self, request):
        b_data = request.data['b_data']
        c_data = request.data['c_data']
        
        # 处理数据并生成部分数据
        a_data = process_data(b_data, c_data)
        
        # 序列化部分数据
        a_serializer = ASerializer(data=a_data)
        if a_serializer.is_valid():
            # 反序列化部分数据并保存在A模型中
            a_instance = a_serializer.save()
            return Response({'id': a_instance.id}, status=status.HTTP_201_CREATED)
        else:
            return Response(a_serializer.errors, status=status.HTTP_400_BAD_REQUEST)

在这个视图中,request.data包含了从请求中传递过来的数据,其中b_datac_data是从B和C模型获取的数据。您可以使用process_data函数来处理这些数据并生成部分数据a_data,然后使用ASerializer类将a_data反序列化为A模型实例。如果反序列化成功,您可以使用a_serializer.save()来保存A模型实例,并返回HTTP 201 Created状态码和A模型实例的ID。如果反序列化失败,您可以返回HTTP 400 Bad Request状态码和错误信息。

至于您的第二个问题,一个serializer可以与多个模型绑定,但是它只能将数据写入一个模型中。如果您想将数据写入多个模型中,您需要为每个模型创建一个单独的serializer类。在视图中,您可以分别使用这些serializer类来处理不同的数据,并将它们保存到不同的模型中。

当然,您也可以使用一个视图和多个serializer来验证不同部分的数据。这可以通过在视图中使用多个serializer类来实现。例如,假设您有一个视图需要验证来自不同位置的数据:

from rest_framework import views, status
from rest_framework.response import Response
from .serializers import BSerializer, CSerializer, DSerializer
from .models import B, C, D

class MyAPIView(views.APIView):
    def post(self, request):
        b_data = request.data['b_data']
        c_data = request.data['c_data']
        d_data = request.data['d_data']
        
        # 验证B模型数据
        b_serializer = BSerializer(data=b_data)
        if not b_serializer.is_valid():
            return Response(b_serializer.errors, status=status.HTTP_400_BAD_REQUEST)
        
        # 验证C模型数据
        c_serializer = CSerializer(data=c_data)
        if not c_serializer.is_valid():
            return Response(c_serializer.errors, status=status.HTTP_400_BAD_REQUEST)
        
        # 验证D模型数据
        d_serializer = DSerializer(data=d_data)
        if not d_serializer.is_valid():
            return Response(d_serializer.errors, status=status.HTTP_400_BAD_REQUEST)
        
        # 将B模型数据写入数据库
        b_instance =b_serializer.save()
        
        # 将C模型数据写入数据库
        c_instance = c_serializer.save()
        
        # 将D模型数据写入数据库
        d_instance = d_serializer.save()
        
        # 返回成功的响应
        return Response({'b_id': b_instance.id, 'c_id': c_instance.id, 'd_id': d_instance.id}, status=status.HTTP_201_CREATED)

在这个视图中,我们使用了三个不同的serializer类(BSerializer,CSerializer和DSerializer)来验证来自不同位置的数据,并将其保存到不同的模型中。如果任何一个serializer验证失败,我们将返回HTTP 400 Bad Request状态码和错误信息。如果所有serializer都验证通过,我们将使用serializer.save()方法将数据写入数据库,并返回HTTP 201 Created状态码和每个模型实例的ID。

serializer接收的输入是模型类,所以需要利用to_internal_value(参数是列表中的一条数据,将 python 类型对象转换成模型类),然后依次调用.is_valid(validate),create(里面有validate) 将数据存入数据库

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,542评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,596评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,021评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,682评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,792评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,985评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,107评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,845评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,299评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,612评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,747评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,441评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,072评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,828评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,069评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,545评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,658评论 2 350

推荐阅读更多精彩内容