使用視圖集ViewSet,能夠將一系列邏輯相關的動做放到一個類中:數據庫
ViewSet視圖集類再也不實現get()、post()等方法,而是實現動做 action 如 list() 、create() 等。post
視圖集只在使用as_view()方法的時候,纔會將action動做與具體請求方式對應上。如:url
class BookInfoViewSet(viewsets.ViewSet): def list(self, request): books = BookInfo.objects.all() serializer = BookInfoSerializer(books, many=True) return Response(serializer.data) def retrieve(self, request, pk=None): try: books = BookInfo.objects.get(id=pk) except BookInfo.DoesNotExist: return Response(status=status.HTTP_404_NOT_FOUND) serializer = BookInfoSerializer(books) return Response(serializer.data)
在設置路由時,咱們能夠以下操做rest
urlpatterns = [ url(r'^books/$', BookInfoViewSet.as_view({'get':'list'}), url(r'^books/(?P<pk>\d+)/$', BookInfoViewSet.as_view({'get': 'retrieve'}) ]
繼承自APIView
與ViewSetMixin
,做用也與APIView基本相似,提供了身份認證、權限校驗、流量管理等。code
ViewSet主要經過繼承ViewSetMixin來實如今調用as_view()時傳入字典(如{'get':'list'})的映射處理工做。對象
在ViewSet中,沒有提供任何動做action方法,須要咱們本身實現action方法。繼承
使用ViewSet一般並不方便,由於list、retrieve、create、update、destory等方法都須要本身編寫,而這些方法與前面講過的Mixin擴展類提供的方法同名,因此咱們能夠經過繼承Mixin擴展類來複用這些方法而無需本身編寫。可是Mixin擴展類依賴與GenericAPIView
,因此還須要繼承GenericAPIView
。路由
GenericViewSet就幫助咱們完成了這樣的繼承工做,繼承自GenericAPIView
與ViewSetMixin
,在實現了調用as_view()時傳入字典(如{'get':'list'}
)的映射處理工做的同時,還提供了GenericAPIView
提供的基礎方法,能夠直接搭配Mixin擴展類使用。get
舉例:io
from rest_framework import mixins from rest_framework.viewsets import GenericViewSet from rest_framework.decorators import action class BookInfoViewSet(mixins.ListModelMixin, mixins.RetrieveModelMixin, GenericViewSet): queryset = BookInfo.objects.all() serializer_class = BookInfoSerializer
url的定義
urlpatterns = [ url(r'^books/$', views.BookInfoViewSet.as_view({'get': 'list'})), url(r'^books/(?P<pk>\d+)/$', views.BookInfoViewSet.as_view({'get': 'retrieve'})), ]
繼承自GenericViewSet
,同時包括了ListModelMixin、RetrieveModelMixin、CreateModelMixin、UpdateModelMixin、DestoryModelMixin。
繼承自GenericViewSet
,同時包括了ListModelMixin、RetrieveModelMixin。
在視圖集中,除了上述默認的方法動做外,還能夠添加自定義動做。
舉例:
from rest_framework import mixins from rest_framework.viewsets import GenericViewSet from rest_framework.decorators import action class BookInfoViewSet(mixins.ListModelMixin, mixins.RetrieveModelMixin, GenericViewSet): queryset = BookInfo.objects.all() serializer_class = BookInfoSerializer def latest(self, request): """ 返回最新的圖書信息 """ book = BookInfo.objects.latest('id') serializer = self.get_serializer(book) return Response(serializer.data) def read(self, request, pk): """ 修改圖書的閱讀量數據 """ book = self.get_object() book.bread = request.data.get('read') book.save() serializer = self.get_serializer(book) return Response(serializer.data)
url的定義
urlpatterns = [ url(r'^books/$', views.BookInfoViewSet.as_view({'get': 'list'})), url(r'^books/latest/$', views.BookInfoViewSet.as_view({'get': 'latest'})), url(r'^books/(?P<pk>\d+)/$', views.BookInfoViewSet.as_view({'get': 'retrieve'})), url(r'^books/(?P<pk>\d+)/read/$', views.BookInfoViewSet.as_view({'put': 'read'})), ]
在視圖集中,咱們能夠經過action對象屬性來獲取當前請求視圖集時的action動做是哪一個。
例如:
from rest_framework.viewsets import ModelViewSet,ReadOnlyModelViewSet from booktest.models import BookInfo from .serializers import BookInfoModelSerializer from rest_framework.response import Response class BookInfoModelViewSet(ModelViewSet): queryset = BookInfo.objects.all() serializer_class = BookInfoModelSerializer def get_top_5(self,request): """獲取評論值最多的5條數據""" # 操做數據庫 print(self.action) # 獲取本次請求的視圖方法名 經過路由訪問到當前方法中.能夠看到本次的action就是請求的方法名