Django Rest Framework - 如何在ModelSerializer中添加自定义字段

Ron*_*Ron 79 python django django-rest-framework

我创建了一个ModelSerializer并且想要添加一个不属于我的模型的自定义字段.

我在这里找到了添加额外字段的说明,并尝试了以下内容:

customField = CharField(source='my_field')
Run Code Online (Sandbox Code Playgroud)

当我添加此字段并调用我的validate()函数时,此字段不是attrdict的一部分.attr包含除额外字段外指定的所有模型字段.所以我在覆盖验证中无法访问此字段,是吗?

当我将此字段添加到字段列表时,如下所示:

class Meta:
    model = Account
    fields = ('myfield1', 'myfield2', 'customField')
Run Code Online (Sandbox Code Playgroud)

然后我得到一个错误,因为customField它不是我的模型的一部分 - 什么是正确的,因为我想只为这个序列化器添加它.

有没有办法添加自定义字段?

Ida*_*aho 62

事实上,有一个解决方案没有触及任何模型.您可以使用SerializerMethodField哪个允许您将任何方法插入序列化程序.

class FooSerializer(ModelSerializer):
    foo = serializers.SerializerMethodField()

    def get_foo(self, obj):
        return "Foo id: %i" % obj.pk
Run Code Online (Sandbox Code Playgroud)

  • 这是最好的答案,api不应该与模型混为一谈 (4认同)
  • 正如OP在[此评论](/sf/ask/1020867151/#comment20355742_14584287)中提到的那样,他们想要一个可写的字段,而不是`SerializerMethodField` (3认同)

Tom*_*tie 55

你正在做正确的事情,除了CharField(和其他类型的字段)是可写字段.

在这种情况下,您只需要一个简单的只读字段,所以只需使用:

customField = Field(source='get_absolute_url')
Run Code Online (Sandbox Code Playgroud)

  • 谢谢,但是我想要一个可写的字段。我传递了一个可以识别我的用户的用户令牌。但是在我的模型中,我拥有用户,而不是令牌。所以我想传递令牌并通过查询将其“转换”为用户对象。 (4认同)

Lin*_*son 14

...为清楚起见,如果您有以下列方式定义的模型方法:

class MyModel(models.Model):
    ...

    def model_method(self):
        return "some_calculated_result"
Run Code Online (Sandbox Code Playgroud)

您可以将调用所述方法的结果添加到序列化程序中,如下所示:

class MyModelSerializer(serializers.ModelSerializer):
    model_method_field = serializers.CharField(source='model_method')
Run Code Online (Sandbox Code Playgroud)

ps由于自定义字段实际上不是模型中的字段,因此您通常希望将其设置为只读,如下所示:

class Meta:
    model = MyModel
    read_only_fields = (
        'model_method_field',
        )
Run Code Online (Sandbox Code Playgroud)

  • 如果我想要它可写怎么办? (2认同)
  • @Csaba:您只需要为附加内容编写自定义保存和删除挂钩:请参阅“方法”下的“保存和删除挂钩”([此处](http://www.django-rest-framework.org/ api-guide/generic-views/#methods)) 您需要编写自定义 `perform_create(self, serializer)`、`perform_update(self, serializer)`、`perform_destroy(self, instance)`。 (2认同)

小智 13

在这里回答你的问题.您应该添加到您的模型帐户:

@property
def my_field(self):
    return None
Run Code Online (Sandbox Code Playgroud)

现在你可以使用:

customField = CharField(source='my_field')
Run Code Online (Sandbox Code Playgroud)

来源:https://stackoverflow.com/a/18396622/3220916

  • 我已经习惯了这种方法,但是对于那些仅用于特定API调用的事物而言,为模型添加额外的代码并不是很好. (5认同)

Dav*_*han 11

我正在寻找一种将可写自定义字段添加到模型序列化器的解决方案。我找到了这个,这个问题的答案中没有涉及到这个。

看来您确实需要编写自己的简单序列化程序。

class PassThroughSerializer(serializers.Field):
    def to_representation(self, instance):
        # This function is for the direction: Instance -> Dict
        # If you only need this, use a ReadOnlyField, or SerializerField
        return None

    def to_internal_value(self, data):
        # This function is for the direction: Dict -> Instance
        # Here you can manipulate the data if you need to.
        return data
Run Code Online (Sandbox Code Playgroud)

现在您可以使用此序列化器将自定义字段添加到模型序列化器

class MyModelSerializer(serializers.ModelSerializer)
    my_custom_field = PassThroughSerializer()

    def create(self, validated_data):
        # now the key 'my_custom_field' is available in validated_data
        ...
        return instance
Run Code Online (Sandbox Code Playgroud)

如果模型MyModel实际上有一个名为的属性my_custom_field,但您想忽略它的验证器,这也适用。


Fra*_*ant 8

为了表明self.author.full_name,我得到了一个错误Field.它适用于ReadOnlyField:

class CommentSerializer(serializers.HyperlinkedModelSerializer):
    author_name = ReadOnlyField(source="author.full_name")
    class Meta:
        model = Comment
        fields = ('url', 'content', 'author_name', 'author')
Run Code Online (Sandbox Code Playgroud)


Gui*_*ent 6

使用最新版本的Django Rest Framework,您需要在模型中创建一个方法,其中包含您要添加的字段的名称.

class Foo(models.Model):
    . . .
    def foo(self):
        return 'stuff'
    . . .

class FooSerializer(ModelSerializer):
    foo = serializers.ReadOnlyField()

    class Meta:
        model = Foo
        fields = ('foo',)
Run Code Online (Sandbox Code Playgroud)


Ari*_*iel 5

在阅读了这里的所有答案后,我的结论是不可能干净利落地做到这一点。你必须玩得很脏,做一些像创建一个 write_only 字段,然后覆盖validateto_representation方法之类的事情。这对我有用:

class FooSerializer(ModelSerializer):

    foo = CharField(write_only=True)

    class Meta:
        model = Foo
        fields = ["foo", ...]

    def validate(self, data):
        foo = data.pop("foo", None)
        # Do what you want with your value
        return super().validate(data)

    def to_representation(self, instance):
        data = super().to_representation(instance)
        data["foo"] = whatever_you_want
        return data
Run Code Online (Sandbox Code Playgroud)