4. 过滤Filtering

对于列表数据可能需要根据字段进行过滤,我们可以通过添加django-fitlter扩展来增强支持。

  1. pip install django-filter

在配置文件setting.py中增加过滤后端的设置:

  1. INSTALLED_APPS = [
  2. ...
  3. 'django_filters', # 需要注册应用,
  4. ]

  5. REST_FRAMEWORK = {
  6. ...
  7. 'DEFAULT_FILTER_BACKENDS': ('django_filters.rest_framework.DjangoFilterBackend',)
  8. }

在视图中添加filter_fields属性,指定可以过滤的字段

  1. class StudentListView(ListAPIView):
  2. queryset = Student.objects.all()
  3. serializer_class = StudentSerializer
  4. filter_fields = ('age', 'sex')

  5. # 127.0.0.1:8000/four/students/?sex=1

5. 排序

对于列表数据,REST framework提供了OrderingFilter过滤器来帮助我们快速指明数据按照指定字段进行排序。

使用方法:

在类视图中设置filter_backends,使用rest_framework.filters.OrderingFilter过滤器,REST framework会在请求的查询字符串参数中检查是否包含了ordering参数,如果包含了ordering参数,则按照ordering参数指明的排序字段对数据集进行排序。

前端可以传递的ordering参数的可选字段值需要在ordering_fields中指明。

示例:

  1.  
  1. INSTALLED_APPS = [
  2. ...
  3. 'django_filters', # 需要注册应用,
  4. ]
  1. #全局配置排序过滤器
  2. REST_FRAMEWORK = {
  3. ...
  4. 'DEFAULT_FILTER_BACKENDS': ('rest_framework.filters.OrderingFilter',),
  5. }
  1. from rest_framework.filters import OrderingFilter
  2. class StudentListView(ListAPIView):
  3. queryset = Student.objects.all()
  4. serializer_class = StudentModelSerializer
  5. #局部配置排序过滤器
  6. filter_backends = [OrderingFilter]
  7. ordering_fields = ('id', 'age')

  8. # 127.0.0.1:8000/books/?ordering=-age
  9. # -id 表示针对id字段进行倒序排序
  10. # id 表示针对id字段进行升序排序

如果需要在过滤以后再次进行排序,则需要两者结合!

  1. INSTALLED_APPS = [
  2. ...
  3. 'django_filters', # 需要注册应用,
  4. ]
  5. #全局配置过滤器
  6. REST_FRAMEWORK = {
  7. ...
  8. 'DEFAULT_FILTER_BACKENDS':('django_filters.rest_framework.DjangoFilterBackend','rest_framework.filters.OrderingFilter'),
  9. }
  1. from rest_framework.generics import ListAPIView
  2. from students.models import Student
  3. from .serializers import StudentModelSerializer
  4. from django_filters.rest_framework import DjangoFilterBackend
  5. from rest_framework.filters import OrderingFilter
  6. class Student3ListView(ListAPIView):
  7. queryset = Student.objects.all()
  8. serializer_class = StudentModelSerializer
  9. filter_fields = ('age', 'sex')
  10. # 因为局部配置会覆盖全局配置,所以需要重新把过滤组件核心类再次声明,
  11. # 否则过滤功能会失效
  12. filter_backends = [OrderingFilter,DjangoFilterBackend]
  13. ordering_fields = ('id', 'age')

6. 分页Pagination

REST framework提供了分页的支持。

我们可以在配置文件中设置全局的分页方式,如:

  1. REST_FRAMEWORK = {
  2. 'DEFAULT_PAGINATION_CLASS': 'rest_framework.pagination.PageNumberPagination',
  3. 'PAGE_SIZE': 100 # 每页数目
  4. }

也可通过自定义Pagination类,来为视图添加不同分页行为。在视图中通过pagination_clas属性来指明。

  1. class LargeResultsSetPagination(PageNumberPagination):
  2. page_size = 1000
  3. page_size_query_param = 'page_size'
  4. max_page_size = 10000
  5. class BookDetailView(RetrieveAPIView):
  6. queryset = BookInfo.objects.all()
  7. serializer_class = BookInfoSerializer
  8. pagination_class = LargeResultsSetPagination

注意:如果在视图内关闭分页功能,只需在视图内设置

  1. pagination_class = None

可选分页器

1) PageNumberPagination

前端访问网址形式:

  1. GET  http://127.0.0.1:8000/students/?page=4

可以在子类中定义的属性:

  • page_size 每页数目

  • page_query_param 前端发送的页数关键字名,默认为"page"

  • page_size_query_param 前端发送的每页数目关键字名,默认为None

  • max_page_size 前端最多能设置的每页数量

  1. # 声明分页的配置类
  1. from rest_framework.pagination import PageNumberPagination
  2. class StandardPageNumberPagination(PageNumberPagination):
  3. # 默认每一页显示的数据量
  4. page_size = 2
  5. # 允许客户端通过get参数来控制每一页的数据量
  6. page_size_query_param = "size"
  7. max_page_size = 10
  8. # 自定义页码的参数名
  9. page_query_param = "p"

  10. class StudentAPIView(ListAPIView):
  11. queryset = Student.objects.all()
  12. serializer_class = StudentModelSerializer
  13. pagination_class = StandardPageNumberPagination
  1. # 127.0.0.1/four/students/?p=1&size=5

2)LimitOffsetPagination

前端访问网址形式:

  1. GET http://127.0.0.1/four/students/?limit=100&offset=400

可以在子类中定义的属性:

  • default_limit 默认限制,默认值与PAGE_SIZE设置一直

  • limit_query_param limit参数名,默认'limit'

  • offset_query_param offset参数名,默认'offset'

  • max_limit 最大limit限制,默认None

  1. from rest_framework.pagination import LimitOffsetPagination
  2. class StandardLimitOffsetPagination(LimitOffsetPagination):
  3. # 默认每一页查询的数据量,类似上面的page_size
  4. default_limit = 2
  5. limit_query_param = "size"
  6. offset_query_param = "start"

  7. class StudentAPIView(ListAPIView):
  8. queryset = Student.objects.all()
  9. serializer_class = StudentModelSerializer
  10. # 调用页码分页类
  11. # pagination_class = StandardPageNumberPagination
  12. # 调用查询偏移分页类
  13. pagination_class = StandardLimitOffsetPagination

7. 异常处理 Exceptions

REST framework提供了异常处理,我们可以自定义异常处理函数。

  1. from rest_framework.views import exception_handler

  2. def custom_exception_handler(exc, context):
  3. # 先调用REST framework默认的异常处理方法获得标准错误响应对象
  4. response = exception_handler(exc, context)

  5. # 在此处补充自定义的异常处理
  6. if response is None:
  7. response.data['status_code'] = response.status_code

  8. return response

在配置文件中声明自定义的异常处理

  1. REST_FRAMEWORK = {
  2. 'EXCEPTION_HANDLER': 'my_project.my_app.utils.custom_exception_handler'
  3. }

如果未声明,会采用默认的方式,如下

rest_frame/settings.py

  1. REST_FRAMEWORK = {
  2. 'EXCEPTION_HANDLER': 'rest_framework.views.exception_handler'
  3. }

例如:

补充上处理关于数据库的异常

  1. from rest_framework.views import exception_handler as drf_exception_handler
  2. from rest_framework import status
  3. from django.db import DatabaseError

  4. def exception_handler(exc, context):
  5. response = drf_exception_handler(exc, context)

  6. if response is None:
  7. view = context['view']
  8. if isinstance(exc, DatabaseError):
  9. print('[%s]: %s' % (view, exc))
  10. response = Response({'detail': '服务器内部错误'}, status=status.HTTP_507_INSUFFICIENT_STORAGE)

  11. return response 

REST framework定义的异常

  • APIException 所有异常的父类

  • ParseError 解析错误

  • AuthenticationFailed 认证失败

  • NotAuthenticated 尚未认证

  • PermissionDenied 权限决绝

  • NotFound 未找到

  • MethodNotAllowed 请求方式不支持

  • NotAcceptable 要获取的数据格式不支持

  • Throttled 超过限流次数

  • ValidationError 校验失败

也就是说,很多的没有在上面列出来的异常,就需要我们在自定义异常中自己处理了。

自定义异常处理

  1. REST_FRAMEWORK = {
  2. 'EXCEPTION_HANDLER': 'my_project.exceptions.custom_exception_handler'
  3. }
  1. class APIError(Exception):
  2. pass
  3.  
  4. class Student2APIView(APIView):
  5. def get(self,request,pk):
  6. try:
  7. instance = Student.objects.get(pk=pk)
  8. except Student.DoesNotExist:
  9. raise APIError("自定义的API错误")
  10. # return Response({"message":"访问的商品已经下架~"})
  11.  
  12. serializer = StudentModelSerializer(instance=instance)
  13. return Response(serializer.data)

views.py

  1. from rest_framework.response import Response
  2. from rest_framework.views import exception_handler
  3.  
  4. from four.views import APIError
  5.  
  6. def custom_exception_handler(exc, context):
  7. # exc 异常对象
  8. # context 异常发生时上下文环境
  9.  
  10. # 先调用REST framework默认的异常处理方法获得标准错误响应对象
  11. response = exception_handler(exc, context)
  12.  
  13. # 在此处补充自定义的异常处理
  14. if response is None:
  15. if isinstance(exc,APIError):
  16. # 记录错误信息,一般记录到文件中,后头开发项目时,我们会使用python的日志功能记录
  17. return Response({"message":"API报错了"})
  18.  
  19. return response

exceptions.py

8. 自动生成接口文档

REST framework可以自动帮助我们生成接口文档。

接口文档以网页的方式呈现。

自动接口文档能生成的是继承自APIView及其子类的视图。

8.1. 安装依赖

REST framewrok生成接口文档需要coreapi库的支持。

  1. pip install coreapi

8.2. 设置接口文档访问路径

在总路由中添加接口文档路径。

文档路由对应的视图配置为rest_framework.documentation.include_docs_urls

参数title为接口文档网站的标题。

  1. from rest_framework.documentation import include_docs_urls

  2. urlpatterns = [
  3. ...
  4. path('docs/', include_docs_urls(title='站点页面标题'))
  5. ]

8.3. 文档描述说明的定义位置

1) 单一方法的视图,可直接使用类视图的文档字符串,如

  1. class BookListView(generics.ListAPIView):
  2. """
  3. 返回所有图书信息.
  4. """

2)包含多个方法的视图,在类视图的文档字符串中,分开方法定义,如

  1. class BookListCreateView(generics.ListCreateAPIView):
  2. """
  3. get:
  4. 返回所有图书信息.

  5. post:
  6. 新建图书.
  7. """

3)对于视图集ViewSet,仍在类视图的文档字符串中封开定义,但是应使用action名称区分,如

  1. class BookInfoViewSet(mixins.ListModelMixin, mixins.RetrieveModelMixin, GenericViewSet):
  2. """
  3. list:
  4. 返回图书列表数据

  5. retrieve:
  6. 返回图书详情数据

  7. latest:
  8. 返回最新的图书数据

  9. read:
  10. 修改图书的阅读量
  11. """

8.4. 访问接口文档网页

浏览器访问 127.0.0.1:8000/docs/,即可看到自动生成的接口文档。

两点说明:

1) 视图集ViewSet中的retrieve名称,在接口文档网站中叫做read

2)参数的Description需要在模型类或序列化器类的字段中以help_text选项定义,如:

  1. class Student(models.Model):
  2. ...
  3. age = models.IntegerField(default=0, verbose_name='年龄', help_text='年龄')
  4. ...
  5.  
  6. class StudentSerializer(serializers.ModelSerializer):
  7. class Meta:
  8. model = Student
  9. fields = "__all__"
  10. extra_kwargs = {
  11. 'age': {
  12. 'required': True,
  13. 'help_text': '年龄'
  14. }
  15. }

9. Xadmin

xadmin是Django的第三方扩展,比使用Django的admin站点更强大也更方便。

文档:https://xadmin.readthedocs.io/en/latest/index.html

9.1. 安装

通过如下命令安装xadmin的最新版

  1. pip install https://codeload.github.com/sshwsfc/xadmin/zip/django2

在配置文件中注册如下应用

  1. INSTALLED_APPS = [
      ...
       'xadmin',
       'crispy_forms',
       'reversion',
      ...
    ]

    # 修改使用中文界面
    LANGUAGE_CODE = 'zh-Hans'

    # 修改时区
    TIME_ZONE = 'Asia/Shanghai'

xadmin有建立自己的数据库模型类,需要进行数据库迁移

  1. python manage.py makemigrations
    python manage.py migrate

在总路由中添加xadmin的路由信息

  1. import xadmin
    xadmin.autodiscover()

    # version模块自动注册需要版本控制的 Model
    from xadmin.plugins import xversion
    xversion.register_models()

    urlpatterns = [
       path(r'xadmin/', xadmin.site.urls),
    ]

如果之前没有创建超级用户,需要创建,如果有了,则可以直接使用之前的。

  1. python manage.py createsuperuser

9.2. 使用

  • xadmin不再使用Django的admin.py,而是需要编写代码在adminx.py文件中。

  • xadmin的站点管理类不用继承admin.ModelAdmin,而是直接继承object即可。

例如:在子应用中创建adminx.py文件。

站点的全局配置

  1. import xadmin
    from xadmin import views

    class BaseSetting(object):
       """xadmin的基本配置"""
       enable_themes = True  # 开启主题切换功能
       use_bootswatch = True

    xadmin.site.register(views.BaseAdminView, BaseSetting)

    class GlobalSettings(object):
       """xadmin的全局配置"""
       site_title = "路飞学城"  # 设置站点标题
       site_footer = "路飞学城有限公司"  # 设置站点的页脚
       menu_style = "accordion"  # 设置菜单折叠

    xadmin.site.register(views.CommAdminView, GlobalSettings)

站点Model管理

xadmin可以使用的页面样式控制基本与Django原生的admin一直。

  • list_display 控制列表展示的字段

    1. list_display = ['id', 'btitle', 'bread', 'bcomment']
  • search_fields 控制可以通过搜索框搜索的字段名称,xadmin使用的是模糊查询

    1. search_fields = ['id','btitle']
  • list_filter 可以进行过滤操作的列,对于分类、性别、状态

    1. list_filter = ['is_delete']
  • ordering 默认排序的字段

  • show_detail_fields 在列表页提供快速显示详情信息

  • list_editable 在列表页可以快速直接编辑的字段

  • refresh_times 指定列表页的定时刷新

    1. refresh_times = [5, 10,30,60] # 设置允许后端管理人员按多长时间(秒)刷新页面
  • list_export 控制列表页导出数据的可选格式

    1. list_export = ('xls', 'xml', 'json')   list_export设置为None来禁用数据导出功能
      list_export_fields = ('id', 'btitle', 'bpub_date')
  • show_bookmarks 控制是否显示书签功能

    1. show_bookmarks = True
  • data_charts 控制显示图表的样式

    1. data_charts = {
             "order_amount": {
               'title': '图书发布日期表',
               "x-field": "bpub_date",
               "y-field": ('btitle',),
               "order": ('id',)
            },
         #   支持生成多个不同的图表
         #   "order_amount": {
         #     'title': '图书发布日期表',
         #     "x-field": "bpub_date",
         #     "y-field": ('btitle',),
         #     "order": ('id',)
         #   },
        }
    • title 控制图标名称

    • x-field 控制x轴字段

    • y-field 控制y轴字段,可以是多个值

    • order 控制默认排序

  • model_icon 控制菜单的图标【图标的设置可以参考bootstrap的图标css名称】

    1. class BookInfoAdmin(object):
        model_icon = 'fa fa-gift'

      xadmin.site.register(models.BookInfo, BookInfodmin)
  • readonly_fields 在编辑页面的只读字段

  • exclude 在编辑页面隐藏的字段

drf07 过滤 排序 分页 异常处理 自动生成接口文档的更多相关文章

  1. day74:drf:drf其他功能:认证/权限/限流/过滤/排序/分页/异常处理&自动生成接口文档

    目录 1.django-admin 2.认证:Authentication 3.权限:Permissions 4.限流:Throttling 5.过滤:Filtering 6.排序:OrderingF ...

  2. [Django REST framework - 自动生成接口文档、分页]

    [Django REST framework - 自动生成接口文档.分页] 自动生成接口文档 # 后端人员写好接口,编写接口文档,给前端人员看,前端人员依照接口文档开发 # 公司里主流 -后端,使用w ...

  3. rbac介绍、自动生成接口文档、jwt介绍与快速签发认证、jwt定制返回格式

    今日内容概要 RBAC 自动生成接口文档 jwt介绍与快速使用 jwt定制返回格式 jwt源码分析 内容详细 1.RBAC(重要) # RBAC 是基于角色的访问控制(Role-Based Acces ...

  4. Spring Boot(九)Swagger2自动生成接口文档和Mock模拟数据

    一.简介 在当下这个前后端分离的技术趋势下,前端工程师过度依赖后端工程师的接口和数据,给开发带来了两大问题: 问题一.后端接口查看难:要怎么调用?参数怎么传递?有几个参数?参数都代表什么含义? 问题二 ...

  5. Spring Boot Swagger2自动生成接口文档

    一.简介 在当下这个前后端分离的技术趋势下,前端工程师过度依赖后端工程师的接口和数据,给开发带来了两大问题: 1.问题一.后端接口查看难:要怎么调用?参数怎么传递?有几个参数?参数都代表什么含义? 2 ...

  6. .net core 使用swagger自动生成接口文档

     前言 swagger是一个api文档自动生动工具,还集成了在线调试. 可以为项目自动生成接口文档, 非常的方便快捷 Swashbuckle.AspNetCore 是一个开源项目,用于生成 ASP.N ...

  7. WebApi使用swagger ui自动生成接口文档

    之前就写到.最近正在使用webapi.这里介绍一个实用的东西swageer ui现在开发都是前后端分开.我们这里是给前端提供api.有时候对于一个api的描述,并不想专门写一份文档.很浪费时间.swa ...

  8. Django框架深入了解_05 (Django中的缓存、Django解决跨域流程(非简单请求,简单请求)、自动生成接口文档)

    一.Django中的缓存: 前戏: 在动态网站中,用户所有的请求,服务器都会去数据库中进行相应的增,删,查,改,渲染模板,执行业务逻辑,最后生成用户看到的页面. 当一个网站的用户访问量很大的时候,每一 ...

  9. Django rest_framework 自动生成接口文档

    自动生成接口文档 REST framework可以自动帮助我们生成接口文档. 接口文档以网页的方式呈现. 自动接口文档能生成的是继承自APIView及其子类的视图. 1. 安装依赖 REST fram ...

随机推荐

  1. 智能家居APP开发

    智能家居APP开发 APP开发技术qq交流群:347072638 前言,随着智能硬件设备的流行,智能家居開始红火,智能家居就是家用电器的智能化.包含智能锁,灯,空调,灯,音箱等等,移动设备通过WIFI ...

  2. cgi程序读取post发送的特殊字符,尤其适合于微信公众平台开发中发送被动消息

    [问题]用c编写cgi程序怎样取出html表单post来的数据? [分析]html表单post来的数据形如username="zhang"&&password=&q ...

  3. 产品架构model

  4. Unable to update auto-refresh reference 'microsoft.codedom.providers.dotnetcompilerplatform.dll'.

    Unable to update auto-refresh reference 'microsoft.codedom.providers.dotnetcompilerplatform.dll'. Ca ...

  5. 【线程安全】—— 单例类双重检查加锁(double-checked locking)

    1. 三个版本单例类的实现 版本1:经典版 public class Singleton { public static Singleton getInstance() { if (instance ...

  6. ADTS结构

    ADTS全称是(Audio Data Transport Stream),是AAC的一种十分常见的传输格式.转载请注明来自:http://www.binkery.com/ ADTS内容及结构一般情况下 ...

  7. JSP-Runoob:JSP 隐式对象

    ylbtech-JSP-Runoob:JSP 隐式对象 1.返回顶部 1. JSP 隐式对象 JSP隐式对象是JSP容器为每个页面提供的Java对象,开发者可以直接使用它们而不用显式声明.JSP隐式对 ...

  8. 5.3QBXT模拟赛

    出题人:钟惠兴 题目名称 讨厌整除的小明 吸血鬼 鱼的感恩 题目类型 传统型 传统型 传统型 题目目录/可执行文件名 ming vamp fool 输入文件名 ming.in vamp.in fool ...

  9. 03-vue实例生命周期和vue-resource

    vue实例的生命周期 什么是生命周期:从Vue实例创建.运行.到销毁期间,总是伴随着各种各样的事件,这些事件,统称为生命周期! 生命周期钩子:就是生命周期事件的别名而已: 生命周期钩子 = 生命周期函 ...

  10. 对“空引用”说bye-bye

    大家可能经常遇到这种情况:当一个对象为null时,调用这个对象的方法或者属性时,就会报错:“Object reference not set to an instance of an object.” ...