视图集(1、路由匹配规则发生变化;2、视图中定义方法不再按照请求方式定义)
视图集分为:
两个基本视图集:
(1)ViewSet,继承自APIView,没有继承拓展类,业务逻辑还得自己去写
(2)GenericViewSet,继承自GenericAPIView,没有继承拓展类,业务逻辑还得自己去写
两个拓展视图集:
(1)ModelViewSet(增删改查都可以实现了),继承了GenericAPIView,继承了五个拓展类
(2)ReadOnlyModelViewSet(实现了获取多个数据对象和获取单一数据对象),继承了GenericAPIView,继承了ListModelMixin,继承了RetrieveModelMixin。
?ViewSet的使用
ViewSet 继承了APIView,所以APIView怎么写的,这块也怎么写,但是方法名可以自定义,并且路由匹配规则需要修改,路由匹配采用键值对方式
from django.shortcuts import render
from testdjango.models import BookInfo
from book_drf.serializer import BookSerializer
from rest_framework.viewsets import ViewSet
from rest_framework.mixins import ListModelMixin, CreateModelMixin, \
RetrieveModelMixin, UpdateModelMixin, DestroyModelMixin
from rest_framework.request import Request
from rest_framework.response import Response
class Books(ViewSet):
# 使用自定义的方法,并且路由也可以自己定义
def list(self, request):
# 查询的时候
print(request.query_params)
books = BookInfo.objects.all()
# 多个对象返回的时候需要加上many=True
ser = BookSerializer(books, many=True)
return Response(ser.data)
def create(self, request):
# 1、获取请求数据
# 2、验证请求数据
# 3、保存数据
# 4、返回结果
# data = request.body.decode() #“{}”
# data_dict = json.loads(data) # 转换成字典 {}
# 使用rest_framework下面的view,request和response就发生了变化.post表单提交
data = request.data
ser = BookSerializer(data=data)
ser.is_valid() # 验证方法,一旦调用这个is_valid()方法,就会调用序列化器里的数据内容是否符合要求,想看下验证结果
# raise_exception=True 的含义是,一旦检出出错误,自动return,不用我们手写return
# print(ser.validated_data) # validated_data 查看验证后的数据
# return JsonResponse(ser.errors)
# 视图通过save() 来调用serialzer序列化器里的crate()方法
ser.save()
# 这个ser.data 是序列化器里的create方法返回的对象,是这个对象里的data
return Response(ser.data)
class Book(ViewSet):
pass
class BookDRFView(ViewSet):
# 对单一数据的处理,可以自定义方法名
def update(self, request, pk):
# 同保存一样的逻辑,前端数据都传给后端,只是在保存数据的时候,改为更新数据
# 1、获取请求数据
# 2、验证请求数据
# 3、更新数据
# 4、返回结果
# data = request.body.decode()
# data_dict = json.loads(data)
data = request.data
# 查询id的信息
try:
book = BookInfo.objects.get(id=pk)
except:
return Response({'error': '错误信息'}, status=400)
# 更新的时候需要多传递一个对象,也就是将要更新的对象,和接收的对象都传递到序列化器中
ser = BookSerializer(book, data=data)
# 验证
ser.is_valid()
# 保存
ser.save()
# 4、返回数据
return Response(ser.data)
from django.urls import path
from django.conf.urls import url
from book_drf import views, apiview_view, viewset_view
urlpatterns = [
# url(r'^book_drf/$', apiview_view.Books.as_view()),
# url(r'^book_drf/(?P<pk>\d+)/$', apiview_view.Book.as_view()),
# url(r'^books_drf/(?P<pk>\d+)/$', apiview_view.BookDRFView.as_view()),
# ViewSet的路由使用方式
url(r'^book_drf/$', viewset_view.Books.as_view({'get': 'list', 'post': 'create'})),
url(r'^book_drf/(?P<pk>\d+)/$', viewset_view.BookDRFView.as_view({'put': 'update'})),
]
这个时候可以发现自定义的名字都是五个拓展类的方法名,如果自定义的方法名不在这五个里面,路由匹配就要这样去写。在获取数值后面加上方法名字
?url(r'^book_drf/(?P<pk>\d+)/lastdata/$', viewset_view.Book.as_view({'get': 'lastdata'}))
def lastdata(self, request, pk):
book = BookInfo.objects.get(id=pk)
ser = BookSerializer(book)
return Response(ser.data)
urlpatterns = [
# url(r'^book_drf/$', apiview_view.Books.as_view()),
# url(r'^book_drf/(?P<pk>\d+)/$', apiview_view.Book.as_view()),
# url(r'^books_drf/(?P<pk>\d+)/$', apiview_view.BookDRFView.as_view()),
# ViewSet的路由使用方式
url(r'^book_drf/$', viewset_view.Books.as_view({'get': 'list', 'post': 'create'})),
url(r'^book_drf/(?P<pk>\d+)/$', viewset_view.BookDRFView.as_view({'put': 'update'})),
url(r'^book_drf/(?P<pk>\d+)/lastdata/$', viewset_view.BookDRFView.as_view({'get': 'lastdata'})),
]
GenericViewSet使用
GenericViewSet继承的是GenericAPIView,所以就是按照GenericAPIView的写法
from django.shortcuts import render
from testdjango.models import BookInfo
from book_drf.serializer import BookSerializer
from rest_framework.viewsets import ViewSet, GenericViewSet
from rest_framework.mixins import ListModelMixin, CreateModelMixin, \
RetrieveModelMixin, UpdateModelMixin, DestroyModelMixin
from rest_framework.request import Request
from rest_framework.response import Response
class Books(GenericViewSet):
queryset = BookInfo.objects.all() # 指定当前类视图使用的查询集数据
serializer_class = BookSerializer # 指定当前类视图使用的序列化器
# 使用自定义的方法,并且路由也可以自己定义
def list(self, request):
# 查询的时候
print(request.query_params)
books = self.get_queryset()
# 多个对象返回的时候需要加上many=True
ser = self.get_serializer(books, many=True)
return Response(ser.data)
def create(self, request):
# 1、获取请求数据
# 2、验证请求数据
# 3、保存数据
# 4、返回结果
# data = request.body.decode() #“{}”
# data_dict = json.loads(data) # 转换成字典 {}
# 使用rest_framework下面的view,request和response就发生了变化.post表单提交
data = request.data
ser = self.get_serializer(data=data)
ser.is_valid() # 验证方法,一旦调用这个is_valid()方法,就会调用序列化器里的数据内容是否符合要求,想看下验证结果
# raise_exception=True 的含义是,一旦检出出错误,自动return,不用我们手写return
# print(ser.validated_data) # validated_data 查看验证后的数据
# return JsonResponse(ser.errors)
# 视图通过save() 来调用serialzer序列化器里的crate()方法
ser.save()
# 这个ser.data 是序列化器里的create方法返回的对象,是这个对象里的data
return Response(ser.data)
class Book(GenericViewSet):
pass
class BookDRFView(GenericViewSet):
queryset = BookInfo.objects.all() # 指定当前类视图使用的查询集数据
serializer_class = BookSerializer # 指定当前类视图使用的序列化器
# 对单一数据的处理,可以自定义方法名
def update(self, request, pk):
# 同保存一样的逻辑,前端数据都传给后端,只是在保存数据的时候,改为更新数据
# 1、获取请求数据
# 2、验证请求数据
# 3、更新数据
# 4、返回结果
# data = request.body.decode()
# data_dict = json.loads(data)
data = request.data
# 查询id的信息
try:
book = self.get_object()
except:
return Response({'error': '错误信息'}, status=400)
# 更新的时候需要多传递一个对象,也就是将要更新的对象,和接收的对象都传递到序列化器中
ser = self.get_serializer(book, data=data)
# 验证
ser.is_valid()
# 保存
ser.save()
# 4、返回数据
return Response(ser.data)
def lastdata(self, request, pk):
book = self.get_object()
ser = self.get_serializer(book)
return Response(ser.data)
路由匹配规则同ViewSet一样
urlpatterns = [
# url(r'^book_drf/$', apiview_view.as_view()),
# url(r'^book_drf/(?P<pk>\d+)/$', apiview_view.as_view()),
# url(r'^books_drf/(?P<pk>\d+)/$', apiview_view.BookDRFView.as_view()),
# ViewSet的路由使用方式
# url(r'^book_drf/$', viewset_view.Books.as_view({'get': 'list', 'post': 'create'})),
# url(r'^book_drf/(?P<pk>\d+)/$', viewset_view.BookDRFView.as_view({'put': 'update'})),
# url(r'^book_drf/(?P<pk>\d+)/lastdata/$', viewset_view.BookDRFView.as_view({'get': 'lastdata'})),
# GenericViewSet的路由使用方式
url(r'^book_drf/$', genericviewset_view.Books.as_view({'get': 'list', 'post': 'create'})),
url(r'^book_drf/(?P<pk>\d+)/$', genericviewset_view.BookDRFView.as_view({'put': 'update'})),
url(r'^book_drf/(?P<pk>\d+)/lastdata/$', genericviewset_view.BookDRFView.as_view({'get': 'lastdata'})),
]
|