Python Django自定义反序列化

Python Django自定义反序列化,python,json,django,django-rest-framework,Python,Json,Django,Django Rest Framework,我有以下django型号: class Person(models.Model): name = models.CharField() location = models.PointField() 我想为这个模型创建一个序列化器/反序列化器。但是,我收到的JSON对象如下所示: { "userList":[ { "username": "Foo", "lat":40.875736, "lon":8.9438

我有以下django型号:

class Person(models.Model):
    name = models.CharField()
    location = models.PointField()
我想为这个模型创建一个序列化器/反序列化器。但是,我收到的JSON对象如下所示:

{
   "userList":[
      {
         "username": "Foo",
         "lat":40.875736,
         "lon":8.94382834,
      },
      {
      "username": "Bar",
      "lat":40.875736,
      "lon":8.94382834,
      }, 
   ]
}
Serialiser

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


class PersonSerializer(serializers.ModelSerializer):
    personList = PersonListSerializer

    class Meta:
        model = Person
是否可以创建自定义序列化程序/反序列化程序来处理此结构,而无需创建其他模型(PersonList)


提前感谢。

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

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

  • 您需要一个字段,该字段从两个单独的字段中获取其值 表示形式中的字段dict并输出两个单独的值。我避开了前一个,对内部值()做了一个自定义
  • 您需要一个不接受并返回列表的
    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)
    

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

    您当前的反序列化方式是什么?更新了问题。我知道我所拥有的不起作用,我有一个想法,为什么,但可以想出如何解决它!添加了DRF标签,因为现在很明显您正在使用restframework(它不是CoreDjango的一部分)!谢谢非常感谢你!就是这样