drf 分页类、异常类、接口文档
一、基本分页
1、写一个分页类,继承 PageNumberPagination
web 用这个多
http://api.example.org/accounts/?page=4
http://api.example.org/accounts/?page=4&page_size=100
from rest_framework.pagination import PageNumberPagination, LimitOffsetPagination, CursorPagination class CommonPageNumberPagination(PageNumberPagination): # 重写几个类属性 page_size = 3 # 每页显示多少条 page_query_param = 'page' # 指定第几页的key值 http://127.0.0.1:8000/books/?page=3 page_size_query_param = 'size' # 可以指定每页显示多少条 size=300000 max_page_size = 5 # 每页最多显示5条
2、views
引入自定义的分页类
from .page import CommonPageNumberPagination class BookView(ViewSetMixin, ListAPIView): queryset = Book.objects.all() serializer_class = BookSerializer pagination_class = CommonPageNumberPagination
3、效果
二、偏移分页
1、LimitOffsetPagination 使用场景
http://api.example.org/accounts/?limit=4 # 从开始取4条
http://api.example.org/accounts/?offset=4&limit=5 从第4条开始,取5条
from rest_framework.pagination import PageNumberPagination, LimitOffsetPagination, CursorPagination class CommonLimitOffsetPagination(LimitOffsetPagination): # http://api.example.org/accounts/?limit=4 # 从开始取4条 # http://api.example.org/accounts/?offset=4&limit=5 从第4条开始,取5条 default_limit = 2 # 默认每页显示2条 limit_query_param = 'limit' # 每页显示多少条的查询条件 offset_query_param = 'offset' # 从第几条开始取数据 max_limit = 5 # limit最多取5条
2、views 引入
from .page import CommonPageNumberPagination,CommonLimitOffsetPagination,CommonCursorPagination class BookView(ViewSetMixin, ListAPIView): queryset = Book.objects.all() serializer_class = BookSerializer pagination_class = CommonLimitOffsetPagination
3、效果
三、游标分页
1、CommonCursorPagination
app上用这个多
只能上一页和下一页,不能指定跳转到中间的某页---》效率高,大数据量用它、
from rest_framework.pagination import PageNumberPagination, LimitOffsetPagination, CursorPagination class CommonCursorPagination(CursorPagination): # app上用这个多 # 只能上一页和下一页,不能指定跳转到中间的某页---》效率高,大数据量用它、 cursor_query_param = 'cursor' # 查询条件,用不到,需要有 page_size = 2 # 每页显示两条 ordering = 'id' # 按id排序 这个必须是表中字段
2、引入略过
3、效果
补充:视图类中配置了分页类,为什么就有分页了?
- pagination_class = CommonCursorPagination # GenericAPIView类属性 -pagination_class是GenericAPIView的类属性:继承APIVIew的视图类,是不能这样配置的----》自己分页 -还需要继承:ListModelMixin---》它的list方法中,在对所有数据做了分页--》 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) -self.paginate_queryset(queryset)是 GenericAPIView 类的方法 - GenericAPIView的 paginate_queryset def paginate_queryset(self, queryset): if self.paginator is None: return None # 不做分页了 # 做分页 return self.paginator.paginate_queryset(queryset, self.request, view=self) -self.paginator 方法包装成了属性,就是咱们分页类的对象 CommonCursorPagination() @property def paginator(self): self._paginator = self.pagination_class() # 咱们配置的分页类 CommonCursorPagination return self._paginator - 调用了 分页类对象的paginate_queryset---》完成真正的分页 分页类对象.paginate_queryset(queryset, self.request, view=self)
四、继承
from rest_framework.mixins import ListModelMixin from rest_framework.generics import GenericAPIView from .serializer import BookSerializer from rest_framework.views import APIView from rest_framework.response import Response from .page import MyPageNumberPagination class BookView(APIView): def get(self, request): ordering = request.query_params.get('ordering') name = request.query_params.get('name') book_list = Book.objects.all() if ordering: book_list = book_list.order_by(ordering) if name: book_list = book_list.filter(name__contains=name) # 加入分页 # page=MyPageNumberPagination().paginate_queryset(book_list, request,self) # 得到分页对象 pagination = MyPageNumberPagination() # 分页对象调用它的paginate_queryset方法 page = pagination.paginate_queryset(book_list, request, self) # 把page序列化 ser = BookSerializer(instance=page, many=True) # 分页类对象调用分页父类中的get_paginated_response方法 return pagination.get_paginated_response(ser.data) # 模仿get_paginated_response中的返回值Response return Response({'code': 100, 'msg': '成功', 'count': pagination.page.paginator.count, 'next': pagination.get_next_link(), 'previous': pagination.get_previous_link(), 'results': ser.data}) ''' 根据源码写自己的分页 # 配置的分页类的对象调用了paginate_queryset(queryset, self.request, view=self) page = self.paginate_queryset(queryset) # 是paginate_queryset的GenericAPIView ##### 重写了这句话 if page is not None: # 把page进行了序列化 serializer = self.get_serializer(page, many=True) # 返回格式 return self.get_paginated_response(serializer.data) # GenericAPIView--》self.paginator.get_paginated_response(data) PageNumberPagination类的get_paginated_response方法 def get_paginated_response(self, data): return Response(OrderedDict([ ('count', self.page.paginator.count), ('next', self.get_next_link()), ('previous', self.get_previous_link()), ('results', data) ])) '''
五、 异常类
1、写一个类文件
from rest_framework.views import exception_handler from rest_framework.response import Response def common_execption(exc, context): res = exception_handler(exc, context) if not res: # 有值,说明drf异常,它处理完了,没有值,是其他异常,我们自己处理 return Response({'code': 999, 'msg': '非drf错误信息是:%s' % str(exc)}) else: return Response({'code': 888, 'msg': 'drf错误信息,错误信息是:' + res.data.get('detail')})
2、在全局配置文件中引入
REST_FRAMEWORK = { 'EXCEPTION_HANDLER': 'app01.exceptions.common_execption' # 异常 }
3、views中故意引发错误
class BookView(ViewSetMixin, APIView): def list(self, request, *args, **kwargs): book_list = Book.objects.all() # raise APIException('这里是主动异常了') prion(asdas) pagination = CommonPageNumberPagination() # 实例化一个对象 page = pagination.paginate_queryset(book_list, request) ser = BookSerializer(instance=page, many=True) return pagination.get_paginated_response(ser.data) # return Response(ser.data)
4、效果
补充:
1 为什么在配置文件中配置了自己写的全局异常处理函数,只要出了异常,它就会走?
# 之前是先执行了res = exception_handler(exc, context),现在要自己做,处理drf异常 # 为什么在配置文件中配置了自己写的全局异常处理函数,只要出了异常,它就会走? -APIView执行流程---》dispatch的 try: except Exception as exc: response = self.handle_exception(exc) #看这里,exc是错误对象 -执行:self.handle_exception(exc)---》self是视图类的对象 -APIView的handle_exception(exc) -handle_exception源码 def handle_exception(self, exc): exception_handler = self.get_exception_handler() # 猜 拿到的就是配置文件中配的那个函数 response = exception_handler(exc, context) #执行这个函数,传了俩参数 return response -self.get_exception_handler()是如何从配置文件中拿出来的 -self.settings.EXCEPTION_HANDLER # 先从项目配置文件中找:key为它的EXCEPTION_HANDLER,如果项目配置文件没有,拿drf内置的配置文件 # 如果你写了全局异常处理函数,配置好了,但是前端还没有返回固定格式,可能的原因是什么? -1 这个错误不是在三大认证和视图类的方法中产生的,之前产生的--》中间件,包装新的request -2 你写的common_exception执行出错了
2 前后端分离了,后端出了异常,我们不想让前端看到,我们需要捕获全局异常,统一返回格式
# drf 源码中已经处理 APIView--->dispatch---> try: # 1 执行三大认证 # 2 执行视图类的方法 except Exception as exc: response = self.handle_exception(exc) -463行左右: # exception_handler就是配置文件中配置的一个函数-->默认的 # 后期自己写了 exception_handler = self.get_exception_handler() response = exception_handler(exc, context) # 默认执行的是:rest_framework.views.exception_handler函数---》只处理了drf的异常 # 咱们处理全局异常的步骤: 1 写一个函数,在内部处理 from rest_framework.views import exception_handler def common_exception_handler(exc,context): res=exception_handler(exc,context) if res: #这次异常是drf的,并且它处理了 # 我们要统一返回格式 return Response({'code':888,'msg':"系统异常(drf异常),请联系系统管理员:%s"%res.data.get('detail',None)}) else: return Response({'code': 999, 'msg': "系统异常(非drf异常),请联系系统管理员:%s" % str(exc)}) 2 配置在配置文件上