10

我是 Django 框架和 Django REST 框架的新手,但我已经运行了基本的设置和实现。当我为单个对象调用域时,它就像一个魅力,例如http://mydomain.com/location/1(其中 1 是主键)。这给了我这样的 JSON 响应:

{"id": 1, "location": "Berlin", "country": 2}

.. 和http://mydomain.com/country/2响应如下:

{"id": 2, "country": "Germany"}

我需要什么: 现在我需要获取多个位置,例如在调用域http://mydomain.com/all_locations/时。我希望得到这样的回应:

[
  {"id": 1, "location": "Berlin", "country": 2},
  {"id": 2, "location": "New York", "country": 4},
  {"id": 3, "location": "Barcelona", "country": 5},
  {"id": 4, "location": "Moscow", "country": 7}
]

这是可选的:在第二步中,当我调用http://mydomain.com/mix_all_locations_countries/时,我希望在一个响应中包含多个国家和地区,例如:

[
  {"locations":
    {"id": 1, "location": "Berlin", "country": 2},
    {"id": 2, "location": "New York", "country": 4},
    {"id": 3, "location": "Barcelona", "country": 5},
    {"id": 4, "location": "Moscow", "country": 7}
  },
  {"countries":
    {"id": 1, "country": "Brazil"}
    {"id": 2, "country": "Germany"},
    {"id": 3, "country": "Portugual"}
    {"id": 4, "country": "USA"},
    {"id": 5, "country": "Spain"},
    {"id": 6, "country": "Italy"}
    {"id": 7, "country": "Russia"}
  }
]

到目前为止,这是我的实现(仅显示位置的实现):

模型.py中:

class Location(models.Model):
    # variable id and pk are always available
    location = models.CharField(max_length=100)
    country = models.ForeignKey("Country")

serializers.py中:

class LocationsSerializer(serializers.ModelSerializer):
    country_id = serializers.Field(source='country.id')

    class Meta:
        model = Location
        fields = (
            'id',
            'location',
            'country_id',
        )

view.py中:

class LocationAPIView(generics.RetrieveAPIView):
    queryset = Location.objects.all()
    serializer_class = LocationSerializer

urls.py中:

url(r'^location/(?P<pk>[0-9]+)/$', views.LocationAPIView.as_view(), name='LocationAPIView')

我尝试了什么:我认为我不需要修改模型和序列化程序,因为它在调用上述域时适用于单个对象。所以我尝试实现一个LocationsViewSetinviews.py并添加一个新的 url in urls.py,但我失败了。知道如何实施吗?也许只是在 LocationAPIView 中定义一个方法并更改定义一个类似于这样的 url:

url(r'^all_locations/$', views.LocationAPIView.get_all_locations(), name='LocationAPIView')

在此先感谢,我将不胜感激。

最好的问候,迈克尔

4

1 回答 1

15

首先,让我们暂时忘记视图集——它们让一些事情变得更简单,但它们也引入了额外的抽象层,我认为你现在不应该关心。

您提到的第一件事是与您当前的详细端点等效的列表端点。您几乎已经掌握了这一点,您只需要在现有视图旁边引入一个额外的视图。

views.py

class LocationListAPIView(generics.ListAPIView):
    queryset = Location.objects.all()
    serializer_class = LocationSerializer

class LocationDetailAPIView(generics.RetrieveAPIView):
    queryset = Location.objects.all()
    serializer_class = LocationSerializer

现在在你的 URLconf 中连接两个视图。

urls.py

url(r'^location/$', views.LocationListAPIView.as_view(), name='location-list'),
url(r'^location/(?P<pk>[0-9]+)/$', views.LocationDetailAPIView.as_view(), name='location-detail')

请注意,我还更改了 URL 名称样式,使其更符合通常的 Django 约定。

接下来,您想要组合位置 + 国家/地区视图。您不仅可以为此使用现有的通用视图,因为它是相当自定义的行为,但是为...编写视图很容易。

views.py

class CombinedAPIView(APIView):
    def get(self, request):
        locations = Location.objects.all()
        countries = Country.objects.all()

        location_serializer = LocationSerializer(locations, many=True)
        country_serializer = CountrySerializer(countries, many=True)

        return Response({
            'countries': country_serializer.data,
            'locations': location_serializer.data
        })

并将视图连接起来。

urls.py

url(r'^combined/$', views.CombinedAPIView.as_view(), name='combined-list')

请注意,您在生成响应时根本不需要使用序列化程序,您也可以提取每个实例上的所有必需字段,在视图本身中显式地为响应构建数据,但这是一种很好的标准方法将模型实例映射到数据字典中。

希望这会给你足够的开始。:)

于 2013-08-29T11:25:54.103 回答