【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
相关推荐
阿冲Runner34 分钟前
创建一个生产可用的线程池
java·后端
写bug写bug43 分钟前
你真的会用枚举吗
java·后端·设计模式
喵手1 小时前
如何利用Java的Stream API提高代码的简洁度和效率?
java·后端·java ee
-Xie-1 小时前
Maven(二)
java·开发语言·maven
IT利刃出鞘2 小时前
Java线程的6种状态和JVM状态打印
java·开发语言·jvm
薛晓刚2 小时前
当MySQL的int不够用了
数据库
SelectDB技术团队2 小时前
Apache Doris 在菜鸟的大规模湖仓业务场景落地实践
数据库·数据仓库·数据分析·apache doris·菜鸟技术
再吃一根胡萝卜2 小时前
使用 squashmigrations 命令优化 Django 迁移文件
python·django
星空下的曙光2 小时前
mysql 命令语法操作篇 数据库约束有哪些 怎么使用
数据库·mysql
小楓12012 小时前
MySQL數據庫開發教學(一) 基本架構
数据库·后端·mysql