Django 自定义反序列化

nun*_*nco 2 python django json django-rest-framework

我有以下 Django 模型:

class Person(models.Model):
    name = models.CharField()
    location = models.PointField()
Run Code Online (Sandbox Code Playgroud)

我想为此模型创建一个序列化器/反序列化器。但是我收到的 JSON 对象如下:

{
   "userList":[
      {
         "username": "Foo",
         "lat":40.875736,
         "lon":8.94382834,
      },
      {
      "username": "Bar",
      "lat":40.875736,
      "lon":8.94382834,
      }, 
   ]
}
Run Code Online (Sandbox Code Playgroud)

串行器

class PersonListSerializer(serializers.PersonSerializer):
    username = serializers.CharField()
    lat = serializers.FloatField()
    lon = serializers.FloatField()


class PersonSerializer(serializers.ModelSerializer):
    personList = PersonListSerializer

    class Meta:
        model = Person
Run Code Online (Sandbox Code Playgroud)

是否可以创建自定义序列化器/反序列化器来处理此结构,而无需创建额外的模型(PersonList)?

提前致谢。

dhk*_*hke 5

嗯,这花了一段时间,这绝对是一次很好的学习经历。

您的问题可以分为两个单独的问题:

  1. 您需要一个字段,该字段从表示字典中的两个单独字段获取其值,并输出两个单独的值。我对前一个退缩了,做了一个定制to_internal_value()
  2. 您需要一个ListSerializer不接受和返回列表的字典,而是一个具有包含实际列表的单个嵌套字段的字典。

您可以在不接触模型的情况下完成此操作。

class ListDictSerializer(serializers.ListSerializer):
    def get_field_name(self):
        if not hasattr(self.Meta, 'field_name'):
            raise ValueError('ListDictSerializer requires defining Meta.field_name overriding get_field_name()')
        return self.Meta.field_name

    def to_internal_value(self, data):
        field_name = self.get_field_name()
        return super(ListDictSerializer, self).to_internal_value(data[field_name])

    def to_representation(self, data):
        field_name = self.get_field_name()
        return ReturnDict({
                field_name: super(ListDictSerializer, self).to_representation(data)
            }, serializer=self
        )

    @property
    def data(self):
       # skip over the ListSerializer to get the real data without the
        # ReturnList
        ret = super(serializers.ListSerializer, self).data
        return ReturnDict(ret, serializer=self)


class PersonListSerializer(ListDictSerializer):
    class Meta:
        field_name = 'userList'


class PersonSerializer(serializers.ModelSerializer):
    class Meta:
        list_serializer_class = PersonListSerializer
        model = Person
        fields = ('username', 'lat', 'lon')

    username = serializers.CharField(source='name')
    lat = serializers.SerializerMethodField(method_name='get_latitude')
    lon = serializers.SerializerMethodField(method_name='get_longitude')

    def get_latitude(self, instance):
        return instance.location.coords[1]

    def get_longitude(self, instance):
        return instance.location.coords[0]

    def to_internal_value(self, data):
        return ReturnDict({
            'name': data.get('username'),
            'location': Point(data['lat'], data['lon']),
        }, serializer=self)
Run Code Online (Sandbox Code Playgroud)

请注意,DRF 允许您以读/写方式导出任意模型属性(不仅是字段)。也就是说,我们可以使用 适当的 getter 和 setter 来定义模型上的@property latand 。lon但是,像 geos 这样的对象Point在创建后是不可变的,因此您无法有选择地在现有对象上设置单个坐标。