【Django-ninja】分页管理器

django ninja通过@paginate装饰器即可进行分页。内置了两个分页管理器LimitOffsetPagination和PageNumberPagination,能够实现基本的分页要求。当内置分页器不满足要求时,可以继承PaginationBase进行扩展自己的分页管理器。

1 使用分页器

python 复制代码
from ninja.pagination import paginate

@api.get('/users', response=List[UserSchema])
@paginate
def list_users(request):
    return User.objects.all()

/api/users?limit=10&offset=0

返回:

python 复制代码
{
"items": [],
"count": 10
}

2 ninja自带分页器

2.1 LimitOffsetPagination

默认paginate使用的就是该分页器。

2.2 PageNumberPagination

python 复制代码
@api.get("/users")
@paginate(PageNumberPagination, page_size=50)
def list_users(...

返回结果包含items和count

python 复制代码
{
"items": [],
"count": 10
}

2.3 自定义分页器

这个自定义分页器定义了Input、Output、paginate_queryset。

实现了page和page_size参数获取数据。page_size可以通过多种方式进行调节。在定义接口的时候设置固定值,或者使用分页器的默认值,或者使用用户的传输参数。

返回结果,新增多个字段,修改默认items为data。

python 复制代码
#!/usr/bin/env python
# coding=utf-8
# @Time    : 2024/2/2 15:40
# @Software: PyCharm
import math
from ninja import Schema
from ninja.pagination import PaginationBase
from typing import List, Any, Optional


class CustomPagination(PaginationBase):

    def __init__(self, page_size: Optional[int] = None, **kwargs):
        """ 如果指定具体值,那么用户参数中的page_size将会失效。
        page_size: 在定义接口是可以设置的每页记录数。

        例如:
        @api.get("/book")
        @paginate(CustomPagination)
        def book_list(request):
            return Book.objects.all()

        @api.get("/book")
        @paginate(CustomPagination, page_size=10)  # 定义接口默认每页数量
        def book_list(request):
            return Book.objects.all()

        """
        self.page_size = page_size
        super().__init__(**kwargs)

    class Input(Schema):
        """ 输入参数,两个参数。
            1. page(必须
            2. page_size(可选)
        """
        page: int
        page_size: int = 5

    class Output(Schema):
        """输出结果。
            1. data 数据集合
            2. total:总记录数
            3. per_page:每页记录数
            4. total_page:总页数
            5. page:当前页
        """
        data: List[Any]  # <--- data
        total: int
        per_page: int
        total_page: int
        page: int

    def paginate_queryset(self, queryset, pagination: Input, **params):
        page = pagination.page
        page_size = self.page_size or pagination.page_size
        start_idx = (page - 1) * page_size
        end_idx = start_idx + page_size
        total_count = queryset.count()
        return {
            'data': queryset[start_idx: end_idx],  # <--- data
            'total': total_count,
            'per_page': page_size,
            'total_page': math.ceil(total_count / page_size),
            'page': page
        }

    items_attribute: str = "data"  # <--- 将默认数据集合的名称从items修改成data
相关推荐
YCY^v^2 分钟前
JeecgBoot 项目运行指南
java·学习
冰暮流星8 分钟前
javascript之二重循环练习
开发语言·javascript·数据库
人间打气筒(Ada)13 分钟前
jenkins基于Pipeline发布项目
java·pipeline·jenkins·流水线·ci·cd·cicd
爬山算法18 分钟前
Hibernate(88)如何在负载测试中使用Hibernate?
java·后端·hibernate
自不量力的A同学22 分钟前
Solon AI v3.9 正式发布:全能 Skill 爆发
java·网络·人工智能
万岳科技系统开发34 分钟前
食堂采购系统源码库存扣减算法与并发控制实现详解
java·前端·数据库·算法
独断万古他化37 分钟前
【Spring 原理】Bean 的作用域与生命周期
java·后端·spring
玄同76538 分钟前
Python 后端三剑客:FastAPI/Flask/Django 对比与 LLM 开发选型指南
人工智能·python·机器学习·自然语言处理·django·flask·fastapi
*小海豚*43 分钟前
在linux服务器上DNS正常,但是java应用调用第三方解析域名报错
java·linux·服务器
冉冰学姐1 小时前
SSM智慧社区管理系统jby69(程序+源码+数据库+调试部署+开发环境)带论文文档1万字以上,文末可获取,系统界面在最后面
数据库·管理系统·智慧社区·ssm 框架