序列化器嵌套序列化器,重写序列化器create和update方法与使用事务保存数据

序列化器嵌套序列化器

规格表序列化器

class SKUSpecificationSerialzier(serializers.ModelSerializer):
    """
        SKU规格表序列化器
    """
    spec_id = serializers.IntegerField()
    option_id = serializers.IntegerField()
    class Meta:
        model = SKUSpecification  # SKUSpecification中sku外键关联了SKU表
        fields = ("spec_id", 'option_id')

商品表序列化器

class SKUSerializer(serializers.ModelSerializer):
  #增加外键的字段 spu_id
= serializers.IntegerField() category_id = serializers.IntegerField() spu = serializers.StringRelatedField(read_only=True) category = serializers.StringRelatedField(read_only=True)   
  #实现序列化器的嵌套
specs
= SKUSpecificationSerialzier(many=True) class Meta: model=SKU fields='__all__'

重写序列化器create和update方法

默认情况下,嵌套串行器是只读的。如果要支持对嵌套序列化器字段的写操作,则需要创建create()和/或update()方法,以明确指定应如何保存子关系。

def create(self, validated_data):
        # 获取规格信息,并从validated_data数据中,删除规格信息数据
        specs_data = validated_data.pop('specs')
        # 保存sku
        sku = SKU.objects.create(**validated_data)
        # 对规格信息进行遍历,来保存商品规格信息
        for spec_data in specs_data:
            SKUSpecification.objects.create(sku=sku, **spec_data)
        # 返回sku
        return sku
    def update(self, instance, validated_data):
    # 获取规格信息,并从validated_data数据中,删除规格信息数据
    specs_data = validated_data.pop('specs')

    # 方案一: 调用父类 ,去实现没有问题的数据更新
    super().update(instance,validated_data)

    # 方案二
    #获取pk
    # pk = self.context['view'].kwargs.get('pk')
    # #更新数据
    # SKU.objects.filter(id=pk).update(**validated_data)

    # 方案三 把父类的源码 拿来
    # for attr, value in validated_data.items():
    #     setattr(instance, attr, value)
    # instance.save()

    # 对规格信息进行遍历,来保存商品规格信息
    for spec_data in specs_data:
        # item {spec_id: "4", option_id: 8}
        SKUSpecification.objects.filter(sku=instance,
                                        spec_id=spec_data.get('spec_id')). \
            update(option_id=spec_data.get('option_id'))

    return instance

使用事务保存数据

def create(self, validated_data):
        # 获取规格信息,并从validated_data数据中,删除规格信息数据
        specs_data = validated_data.pop('specs')

        with transaction.atomic():
            # 开启事务
            savepoint = transaction.savepoint()
            # 保存sku
            sku = SKU.objects.create(**validated_data)
            # 对规格信息进行遍历,来保存商品规格信息
            for spec_data in specs_data:
                SKUSpecification.objects.create(sku=sku, **spec_data)
            # 清除保存点
            transaction.savepoint_commit(savepoint)
            # 返回sku
            return sku

 

posted @ 2020-05-27 09:47  Tracydzf  阅读(1174)  评论(0编辑  收藏  举报