1.框架一(继承APIView)
这里的第一部分使用骨架请参考我的博客(第三篇),它采用了restframework中最基础的办法(APIView)实现了相关请求,以下的框架都是基于它的
2.框架二(继承ViewSetMixin)
对于框架一,我们只继承APIView,也能实现增删改查的方法,但是不要忘了:
对于查看(get),我们可以查看全部,也能查看局部(添加id)
对于删除,我们也要根据指定id删除,对于修改,我们也要根据指定id修改
那么,我们就要写两个路由,两个类:一个类(CoursesView)中包含了get和post方法,每个方法不用携带id;一个类(CourseDetailView)中包含了get,put和delete三个方法,都需要携带id
这样我们操作的的类过多,每一个都要写两套类,那样工作量就会很大
而框架二是在继承APIView的基础下,再继承ViewSetMixin的类,来实现一套路由的管理
我们先来查看ViewSetMixin的文档说明:
"""
This is the magic.
Overrides `.as_view()` so that it takes an `actions` keyword that performs
the binding of HTTP methods to actions on the Resource.
For example, to create a concrete view binding the 'GET' and 'POST' methods
to the 'list' and 'create' actions...
view = MyViewSet.as_view({'get': 'list', 'post': 'create'})
"""
基于源码我们来重写views下的course.py
from rest_framework.views import APIView
from rest_framework.viewsets import ViewSetMixin
class CoursesView(ViewSetMixin,APIView):
# 查看
def list(self, request, *args, **kwargs):
pass
# 增加
def create(self, request, *args, **kwargs):
pass
# 局部查看
def retrieve(self, request, pk, *args, **kwargs):
pass
# 指定id修改
def update(self, request, pk, *args, **kwargs):
pass
# 指定id删除
def destroy(self, request, pk, *args, **kwargs):
pass
api下的urls.py
from django.conf.urls import url
from api.views import course
urlpatterns = [
url(r'courses/$',course.CoursesView.as_view({'get':'list','post':'create'})),
url(r'courses/(?P<pk>\d+)/$',course.CoursesView.as_view({'get':'retrieve','put':'update','delete':'destroy'}))
]
实际情况下,我们也不会这几种方法都用到,主要还是查看
3.框架三(继承ViewSetMixin和generics.GenericAPIView)
相关源码说明:
基于源码我们来重写views下的course.py
from api import models
from rest_framework.response import Response
from api.serializers.course import CourseSerializer
from rest_framework.viewsets import GenericViewSet
from rest_framework.mixins import ListModelMixin,CreateModelMixin,\
RetrieveModelMixin,UpdateModelMixin,DestroyModelMixin,ListModelMixin
# 这里我只使用了list方法,你要是用其他哪一种方法,直接把它的相关类写进即可
class CoursesView(ListModelMixin,GenericViewSet):
queryset = models.Course.objects.all()
def list(self, request, *args, **kwargs):
course_list = models.Course.objects.all()
ser = CourseSerializer(instance=course_list, many=True)
return Response(ser.data)
api下的urls.py
from django.conf.urls import url
from api.views import course
urlpatterns = [
url(r'courses/$',course.CoursesView.as_view({'get':'list'})),
url(r'courses/(?P<pk>\d+)/$',course.CoursesView.as_view({'get':'retrieve'}))
]
serializers下的course.py(也没有做改变)
from rest_framework import serializers
class CourseSerializer(serializers.Serializer):
id = serializers.IntegerField()
name = serializers.CharField()
效果:
3.1基于框架三的扩展(可以使用序列化组件ModelSerializer,它继承了Serializer)
它的相关源码:
这里相关代码我就不演示了,可以查看之前的代码学习相关知识
相关说明:对于框架一和框架二适用于用户请求处理时业务逻辑复杂的情况。
第三种框架,由于渲染器内部会调用视图对象的get_queryset方法,所以必须要设置queryset字段。
针对第三种解决方案:
1. 设置queryset字段
class CoursesView(ListModelMixin,GenericViewSet):
queryset = models.Course.objects.all()
def list(self, request, *args, **kwargs):
course_list = models.Course.objects.all()
ser = CourseSerializer(instance=course_list, many=True)
return Response(ser.data)
2. 不使用渲染器,没有那种好看的面板渲染了
class CoursesView(ListModelMixin,GenericViewSet):
renderer_classes = [JSONRenderer,]
def list(self, request, *args, **kwargs):
course_list = models.Course.objects.all()
ser = CourseSerializer(instance=course_list, many=True)
return Response(ser.data)