2017-07-28 144 views
1

我只是在学习geo-django。我可以从一个点找出所有地方的距离。但是当我使用.values方法的注释distance场,我得到GeoDjango:距离对象不可序列化

TypeError: Object of type 'Distance' is not JSON serializable

这里是我的代码片段

#models.py 
import uuid 
from django.contrib.gis.db import models 
from django.contrib.gis.db.models.functions import Distance 
from django.contrib.gis.geos import Point 
class PlaceManager(models.GeoManager): 
    def get_queryset(self): 
     qs = super(PlaceManager, self).get_queryset() 
     qs = qs.annotate(
     latitude=models.ExpressionWrapper(models.Func('position', function='ST_X'), output_field=models.FloatField()), 
     longitude=models.ExpressionWrapper(models.Func('position', function='ST_Y'), output_field=models.FloatField()), 
    ) 
    return qs.distinct() 

    def nearby_places(self, lat, lng): 
     p = Point(lat, lng, srid=4326) 
     qs = self.get_queryset() 
     qs = qs.annotate(
      distance=Distance('position', p) 
     ) 
     return qs.order_by('distance').distinct() 


class Place(models.Model): 
    id = models.UUIDField(default=uuid.uuid4, editable=False, primary_key=True, db_index=True) 
    position = models.PointField() 
    address = models.TextField(default=None, null=True, blank=True) 
    objects = PlaceManager() 

    def __str__(self): 
     return '{},{}'.format(self.position.x, self.position.y) 

现在代码片断我是这样

from rest_framework.views import APIView 
from rest_framework import status 
from rest_framework.response import Response 

class NearbyPlaces(APIView): 
    def get(self, request): 
     p = Place.objects.nearby_places(30.45, -90.43) 
     p = p.values('distance', 'address', 'latitude', 'longitude') 
     return Response(p, status=status.HTTP_200_OK) 

这里的价值p就是这样的

<GeoQuerySet [{'distance': Distance(m=7596021.71574835), 'address': 'New York City, New York','latitude': 13.4586, 'longitude': 45.6789}]> 

因此,所有我需要的是在这里,而不是'distance': 7596021.71574835'distance': Distance(m=7596021.71574835)

任何帮助?提前致谢。

+0

我建议不使用.values(),而是使用一个串行器。这样,当serialiser获取查询集时,您可以选取距离值并将其设置为值。 http://www.django-rest-framework.org/tutorial/1-serialization/#creating-a-serializer-class –

+0

那么找到了另一种方式来做到这一点。只需制作一个渲染器,并将其用作'django rest框架中的默认渲染器类。“# –

+0

在**#models.py **中,只需将** distance = Distance('position',p)**更改为** distance =距离('位置',p).m **,这将起作用。这样,您可以保留所有其他代码原来的**,而无需使用Renderer类。 –

回答

1

刚刚找到一种方法来做到这一点。 不得不在rest框架中创建一个Renderer类,并在那里处理Distance对象。代码片段是这样的。

encoders.py

from rest_framework.utils.encoders import JSONEncoder 
from django.contrib.gis.measure import Distance 


class CustomJsonEncoderWithDistance(JSONEncoder): 
    def default(self, obj): 
     if isinstance(obj, Distance): 
      print(obj) 
      return obj.m 
     return super(CustomJsonEncoderWithDistance, self).default(obj) 

renders.py

from rest_framework.renderers import JSONRenderer 
from .encoders import CustomJsonEncoderWithDistance 


class CustomJsonRenderer(JSONRenderer): 
    encoder_class = CustomJsonEncoderWithDistance 

settings.py

... 
REST_FRAMEWORK = { 
    'DEFAULT_RENDERER_CLASSES': (
     'app.renderers.CustomJsonRenderer', 
    ) 
}