drf04 drf视图类

REST framework 提供了众多的通用视图基类与扩展类,以简化视图的编写。

1、2个视图基类

1.1、 APIView

rest_framework.views.APIView
APIViewView
APIViewView
RequestHttpRequesetResponseAPIException

支持定义的类属性

  • authentication_classes 列表或元祖,身份认证类

  • permissoin_classes 列表或元祖,权限检查类

  • throttle_classes 列表或元祖,流量控制类

APIView

举例:

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

# url(r'^students/$', views.StudentsAPIView.as_view()),
class StudentsAPIView(APIView):
   def get(self, request):
       data_list = Student.objects.all()
       serializer = StudentModelSerializer(instance=data_list, many=True)
       return Response(serializer.data)

1.2、 GenericAPIView[通用视图类]

rest_framework.generics.GenericAPIView
APIVIew

提供的关于序列化器使用的属性与方法

def get_serializer_class(self):
   if self.request.user.is_staff:
       return FullAccountSerializer
   return BasicAccountSerializer

提供的关于数据库查询的属性与方法

def get_queryset(self):
   user = self.request.user
   return user.accounts.all()
# url(r'^books/(?P<pk>\d+)/$', views.BookDetailView.as_view()),
class BookDetailView(GenericAPIView):
   queryset = BookInfo.objects.all()
   serializer_class = BookInfoSerializer

   def get(self, request, pk):
       book = self.get_object() # get_object()方法根据pk参数查找queryset中的数据对象
       serializer = self.get_serializer(book)
       return Response(serializer.data)

其他可以设置的属性

  • pagination_class 指明分页控制类

  • filter_backends 指明过滤控制后端

为了方便学习上面的GenericAPIView通用视图类,我们新建一个子应用。

python manage.py startapp gen

2、 5个视图扩展类

作用:

提供了几种后端视图(对数据资源进行曾删改查)处理流程的实现,如果需要编写的视图属于这五种,则视图可以通过继承相应的扩展类来复用代码,减少自己编写的代码量。

这五个扩展类需要搭配GenericAPIView父类,因为五个扩展类的实现需要调用GenericAPIView提供的序列化器与数据库查询的方法。

1)ListModelMixin

list(request, *args, **kwargs)

该Mixin的list方法会对数据进行过滤和分页。

源代码:

class ListModelMixin(object):
   """
  List a queryset.
  """
   def list(self, request, *args, **kwargs):
       # 过滤
       queryset = self.filter_queryset(self.get_queryset())
       # 分页
       page = self.paginate_queryset(queryset)
       if page is not None:
           serializer = self.get_serializer(page, many=True)
           return self.get_paginated_response(serializer.data)
       # 序列化
       serializer = self.get_serializer(queryset, many=True)
       return Response(serializer.data)

举例:

from rest_framework.mixins import ListModelMixin

class BookListView(ListModelMixin, GenericAPIView):
   queryset = BookInfo.objects.all()
   serializer_class = BookInfoSerializer

   def get(self, request):
       return self.list(request)

2)CreateModelMixin

create(request, *args, **kwargs)

如果序列化器对前端发送的数据验证失败,返回400错误。

源代码:

class CreateModelMixin(object):
   """
  Create a model instance.
  """
   def create(self, request, *args, **kwargs):
       # 获取序列化器
       serializer = self.get_serializer(data=request.data)
       # 验证
       serializer.is_valid(raise_exception=True)
       # 保存
       self.perform_create(serializer)
       headers = self.get_success_headers(serializer.data)
       return Response(serializer.data, status=status.HTTP_201_CREATED, headers=headers)

   def perform_create(self, serializer):
       serializer.save()

   def get_success_headers(self, data):
       try:
           return {'Location': str(data[api_settings.URL_FIELD_NAME])}
       except (TypeError, KeyError):
           return {}

3)RetrieveModelMixin

retrieve(request, *args, **kwargs)

如果存在,返回200, 否则返回404。

源代码:

class RetrieveModelMixin(object):
   """
  Retrieve a model instance.
  """
   def retrieve(self, request, *args, **kwargs):
       # 获取对象,会检查对象的权限
       instance = self.get_object()
       # 序列化
       serializer = self.get_serializer(instance)
       return Response(serializer.data)

举例:

class BookDetailView(RetrieveModelMixin, GenericAPIView):
   queryset = BookInfo.objects.all()
   serializer_class = BookInfoSerializer

   def get(self, request, pk):
       return self.retrieve(request)

4)UpdateModelMixin

update(request, *args, **kwargs)
partial_update(request, *args, **kwargs)

成功返回200,序列化器校验数据失败时,返回400错误。

源代码:

class UpdateModelMixin(object):
   """
  Update a model instance.
  """
   def update(self, request, *args, **kwargs):
       partial = kwargs.pop('partial', False)
       instance = self.get_object()
       serializer = self.get_serializer(instance, data=request.data, partial=partial)
       serializer.is_valid(raise_exception=True)
       self.perform_update(serializer)

       if getattr(instance, '_prefetched_objects_cache', None):
           # If 'prefetch_related' has been applied to a queryset, we need to
           # forcibly invalidate the prefetch cache on the instance.
           instance._prefetched_objects_cache = {}

       return Response(serializer.data)

   def perform_update(self, serializer):
       serializer.save()

   def partial_update(self, request, *args, **kwargs):
       kwargs['partial'] = True
       return self.update(request, *args, **kwargs)

5)DestroyModelMixin

destroy(request, *args, **kwargs)

成功返回204,不存在返回404。

源代码:

class DestroyModelMixin(object):
   """
  Destroy a model instance.
  """
   def destroy(self, request, *args, **kwargs):
       instance = self.get_object()
       self.perform_destroy(instance)
       return Response(status=status.HTTP_204_NO_CONTENT)

   def perform_destroy(self, instance):
       instance.delete()

使用GenericAPIView和视图扩展类,实现api接口,代码:

其它代码以上面的例子类似!

3、 GenericAPIView的视图子类

1)CreateAPIView

提供 post 方法

继承自: GenericAPIView、CreateModelMixin

2)ListAPIView

提供 get 方法

继承自:GenericAPIView、ListModelMixin

3)RetrieveAPIView

提供 get 方法

继承自: GenericAPIView、RetrieveModelMixin

4)DestoryAPIView

提供 delete 方法

继承自:GenericAPIView、DestoryModelMixin

5)UpdateAPIView

提供 put 和 patch 方法

继承自:GenericAPIView、UpdateModelMixin

6)RetrieveUpdateAPIView

提供 get、put、patch方法

继承自: GenericAPIView、RetrieveModelMixin、UpdateModelMixin

7)RetrieveUpdateDestoryAPIView

提供 get、put、patch、delete方法

继承自:GenericAPIView、RetrieveModelMixin、UpdateModelMixin、DestoryModelMixin

其它代码以上面的例子类似!
 

4、 常用视图集父类(ViewSet、 GenericViewSet、 ModelViewSet、 ReadOnlyModelViewSet)

1) ViewSet

APIViewViewSetMixin

ViewSet主要通过继承ViewSetMixin来实现在调用as_view()时传入字典(如{'get':'list'})的映射处理工作。

在ViewSet中,没有提供任何动作action方法,需要我们自己实现action方法。

使用视图集ViewSet,可以将一系列逻辑相关的动作放到一个类中:

  • list() 提供一组数据

  • retrieve() 提供单个数据

  • create() 创建数据

  • update() 保存数据

  • destory() 删除数据

ViewSet视图集类不再实现get()、post()等方法,而是实现动作 action 如 list() 、create() 等。

视图集只在使用as_view()方法的时候,才会将action动作与具体请求方式对应上。如:

2)GenericViewSet

GenericAPIViewGenericAPIView
GenericAPIViewViewSetMixin{'get':'list'}GenericAPIView

举例:

from rest_framework.viewsets import GenericViewSet
from rest_framework.mixins import ListModelMixin,CreateModelMixin,RetrieveModelMixin,UpdateModelMixin,DestroyModelMixin
class Student4ViewSet(GenericViewSet,ListModelMixin,CreateModelMixin,RetrieveModelMixin,UpdateModelMixin,DestroyModelMixin):
   queryset = Student.objects.all()
   serializer_class = StudentModelSerializer

url的定义

urlpatterns = [
   path("students7/", views.Student4ViewSet.as_view({"get": "list", "post": "create"})),
   re_path("students7/(?P<pk>\d+)/", views.Student4ViewSet.as_view({"get": "retrieve","put":"update","delete":"destroy"})),

]

3)ModelViewSet

GenericViewSet

4)ReadOnlyModelViewSet

GenericViewSet

1.3.2 视图集中定义附加action动作

在视图集中,除了上述默认的方法动作外,还可以添加自定义动作。

举例:

from rest_framework.viewsets import ModelViewSet,ReadOnlyModelViewSet
class StudentModelViewSet(ModelViewSet):
   queryset = Student.objects.all()
   serializer_class = StudentModelSerializer

   def login(self,request):
       """学生登录功能"""
       return Response({"message":"登录成功"})

url的定义

urlpatterns = [
   path("students8/", views.StudentModelViewSet.as_view({"get": "list", "post": "create"})),
   re_path("students8/(?P<pk>\d+)/",
           views.StudentModelViewSet.as_view({"get": "retrieve", "put": "update", "delete": "destroy"})),

   path("stu/login/",views.StudentModelViewSet.as_view({"get":"login"}))

]

1.3.3 action属性

在视图集中,我们可以通过action对象属性来获取当前请求视图集时的action动作是哪个。

例如:

from rest_framework.viewsets import ModelViewSet
from students.models import Student
from .serializers import StudentModelSerializer
from rest_framework.response import Response
class StudentModelViewSet(ModelViewSet):
   queryset = Student.objects.all()
   serializer_class = StudentModelSerializer

   def get_new_5(self,request):
       """获取最近添加的5个学生信息"""
       # 操作数据库
       print(self.action) # 获取本次请求的视图方法名
       
       
通过路由访问到当前方法中.可以看到本次的action就是请求的方法名