Python 在DRF 3中的ModelSerializer上添加非模型字段
如何在DRF 3中的ModelSerializer上添加非模型字段?i、 e.添加实际模型上不存在的字段Python 在DRF 3中的ModelSerializer上添加非模型字段,python,django,django-rest-framework,Python,Django,Django Rest Framework,如何在DRF 3中的ModelSerializer上添加非模型字段?i、 e.添加实际模型上不存在的字段 class TestSerializer(serializers.ModelSerializer): url = serializers.HyperlinkedIdentityField(view_name='vote_detail') non_field = serializers.CharField() # no corresponding model property
class TestSerializer(serializers.ModelSerializer):
url = serializers.HyperlinkedIdentityField(view_name='vote_detail')
non_field = serializers.CharField() # no corresponding model property.
class Meta:
model = vote_model
fields = ("url", "non_field")
def create(self, validated_data):
print(direction=validated_data['non_field'])
但DRF 3给了我一个错误:
Got AttributeError when attempting to get a value for field `non_field` on serializer `TestSerializer`.
The serializer field might be named incorrectly and not match any attribute or key on the `Test` instance.
Original exception text was: 'Test' object has no attribute 'non_field'.
我已经搜索了堆栈并找到了一些解决方案,但这些是指我使用DRF3的DRF2。在这个版本上有解决这个问题的方法吗?举个例子可能会对您有所帮助
class ExtensibleModelSerializerOptions(serializers.SerializerOptions):
"""
Meta class options for ModelSerializer
"""
def __init__(self, meta):
super(ExtensibleModelSerializerOptions, self).__init__(meta)
self.model = getattr(meta, 'model', None)
self.read_only_fields = getattr(meta, 'read_only_fields', ())
self.non_native_fields = getattr(meta, 'non_native_fields', ())
class ExtensibleModelSerializer(serializers.ModelSerializer):
_options_class = ExtensibleModelSerializerOptions
def restore_object(self, attrs, instance=None):
"""
Deserialize a dictionary of attributes into an object instance.
You should override this method to control how deserialized objects
are instantiated.
"""
for field in self.opts.non_native_fields:
attrs.pop(field)
return super(ExtensibleModelSerializer, self).restore_object(attrs, instance)
资料来源:
还是经历这一切
资料来源:
serializers.CharField(write_only=True)
和serializers.ListField(…)
是一个很好的解决方案,可以将额外的数据提供给.create()
和.update()
方法,作为单个字符串或字符串列表(可以将ListField
与其他序列化程序字段类型混合使用)。使用此方法,还可以定义
def validate\u write\u only\u char\u字段
,以实现一些快速而简单的验证
serializers.SerializerMethodField()
允许您从序列化程序上定义的方法向序列化程序输出中添加一些自定义只读字段
read\u only\u custom\u model\u字段
将使用模型上的方法来读取一些数据,严格来说不是模型字段,而是自定义方法。即
class MyModel(models.Model):
my_field = models.CharField(max_length=100)
@property
def custom_property(self):
return "Perform calculations, combine with related models, etc. etc."
如前所述,有两种方法。(1) 添加模型属性。(2) 添加模型字段。我觉得在模型中添加@属性在这篇文章中得到了很好的解释。如果你想让你的模型“精干而中庸”,使用一个方法字段。尽管Chandus的回答忽略了一些关键点:
class DeliveryItemSerializer(serializers.ModelSerializer):
product_name = serializers.SerializerMethodField(read_only=True)
def get_product_name(self, obj):
return obj.product.name
class Meta:
model = DeliveryItem
fields = (
(...your field names),
'product_name',)
get\u
field\u name。如果使用其他名称,请使用SerializerMethodField()
@chandu那么这个问题在3.0+中还没有解决吗?或者现在有解决方法了吗?你可以使用这个方法或者下面的方法。我在模型上使用了一个“`@property``解决了这个问题,这是你在上面发布的答案。你能发布并回答我的问题吗?non_field=serializers.serializers.SerializerMethodField()#没有相应的模型属性-删除额外的序列化程序?我试图做完全相同的事情,但我有一个错误告诉我,我的模型没有这个字段。有人找到解决办法了吗?这个答案是错误的。在上面的链接中,它说的第一件事是SerializerMethodField是一个只读字段。感谢它的工作,因为我们将它添加到
字段
列表中。这是最完整的答案,完美~也是我找到的唯一一个(为时已晚)关于只写非模型字段的答案。您最终可以在validate
方法中弹出键,可能更合适。非常感谢@ARJMPwrite_only
正是我所需要的。谢谢
class MySerializer(serializers.ModelSerializer):
write_only_char_field = serializers.CharField(write_only=True)
write_only_list_char_field = serializers.ListField(child=serializers.CharField(max_length=100, default=''), write_only=True)
empty_method_field = serializers.SerializerMethodField()
read_only_custom_model_field = serializers.CharField(source='custom_property', read_only=True)
def create(self, validated_data):
validated_data.pop('write_only_char_field', None)
validated_data.pop('write_only_list_char_field', None)
return super().create(validated_data)
class MyModel(models.Model):
my_field = models.CharField(max_length=100)
@property
def custom_property(self):
return "Perform calculations, combine with related models, etc. etc."
class DeliveryItemSerializer(serializers.ModelSerializer):
product_name = serializers.SerializerMethodField(read_only=True)
def get_product_name(self, obj):
return obj.product.name
class Meta:
model = DeliveryItem
fields = (
(...your field names),
'product_name',)