美文网首页
Django 自定义分页类和使用总结

Django 自定义分页类和使用总结

作者: 菩提老鹰 | 来源:发表于2023-07-05 16:19 被阅读0次

一、关于为何要分页

当处理大量数据时,如果一次将这些数据查询出来进行响应,必然对服务器内存、负载有所影响,影响低接口响应,进而影响用户体验。
常见的方式是将数据分段展示给用户,如果当前分段中没有需要的数据,那么就可以通过翻页或者指定具体的页码来查看更多的数据。

Django提供了默认的分页功能,同时也允许自定义分页来满足特定需求。

二、Django的默认分页 (函数视图)

在视图中,使用Django提供的Paginator类来分页数据。一般是和Django的Template结合使用

1、需要引入Paginator和要分页的数据模型。

from django.core.paginator import Paginator
from myapp.models import Post

2、获取要分页的数据,并创建 Paginator 对象,进行分页设置

# 获取数据
posts = Post.objects.all()

# 创建Paginator对象
# 作为测试这里 per_page 可以小一些
paginator = Paginator(posts, per_page=2)

3、或者指定的页码 数据, 默认习惯使用 page 参数

# 或者当前页码 
page_number = request.GET.get('page', 1)

# 获取 页码对应的数据
page_data = paginator.get_page(page_number)

# 数据渲染到 template模板中去
return render(request, 'posts/list.html', {'page_data': page_data})

4、在 Template模板中使用page_data 来展示数据和分页导航


<ul>
    {% for post in page_data %}
        <li> {{ post.title }}</li>
    {% endfor %}
</ul>

<br>

<!-- 分页导航 -->
<div class='pagination'>
    {% if page_data.has_previous %}
        <span> 
            <a href="?page=1">&laquo;  首页</a>
            <a href="?page={{ page_data.previous_page_number }}">前一页</a>
        </span>
    {% endif %}
    当前页码: {{ page_data.number }} / {{ page_data.paginator.count }}
    {% if page_data.has_next %}
        <span>
            <a href="?page={{ page_data.next_page_number }}">后一页</a>
            <a href="?page={{ page_data.paginator.num_pages }}">&laquo;  尾页</a>
        </span>
    {% endif %}
</div>

实际的效果如下图所示:

Django Template 分页实现demo演示

三、Django的默认分页(类视图)

上面介绍的是在Django的函数视图种如何使用分页功能, 这趴介绍在类视图中使用分页功能。

如果对Django 函数视图和类视图有过了解的同学就会很容易知道。如何把上面的 函数视图改造成 类视图。

其实很简单,类视图中的get()方法的代码逻辑就是上面的 函数视图的代码。如下


class PostListViewDemo(View):
    def get(self, request):
         # 获取数据
        posts = Post.objects.all()

        # 创建Paginator对象
        paginator = Paginator(posts, per_page=3)

        # 或者当前页码 
        page_number = request.GET.get('page', 1)

        # 获取 页码对应的数据
        page_data = paginator.get_page(page_number)
        # 这里输出debug日志,
        print(page_data, type(page_data), dir(page_data))

        # 数据渲染到 template模板中去
        return render(request, 'posts/list2.html', {'page_data': page_data})

需要主要的就是在urls中设置路由时。 类视图需要使用 as_view() 方法,如下

from django.urls import path 

from appdemo.views import post_list, PostListViewDemo

urlpatterns = [
    path('list1/', post_list, name='post-list-1'),
    # 注意这里
    path('list2/', PostListViewDemo.as_view(), name='post-list-2'),
]

四、更简单的Django类视图分页实现

我们知道只有在大数据量,也就是 列举数据的时候用到分页。

Django在 django.views.generic 中提供了更简便的封装类 ListView

上面的分页功能可以很简单的如下实现

from django.views.generic import ListView

from appdemo.models import Post


class PostListView(ListView):
    paginate_by = 4
    model = Post
    # 注意 📢
    # 如果这里不指定 template_name, 那么默认的值为  appdemo/post_list.html
    # 组成是 app_label, model_name, template_name_suffix(对于ListView而言是 _list.html)
    template_name = "posts/list3.html"

这里需要注意的是:

在Template文件 posts/list3.html 不能使用前面两种方式中 自定义的 page_data 对象 ,而要使用固定格式的 page_obj 对象。

具体餐参考 Django源码 django.views.generic.list.MultipleObjectMixin 类中的get_context_data() 方法的返回

源码解释

因为 ListView 继承自 MultipleObjectTemplateResponseMixinBaseListView , 在 BaseListView 中定义了 get() 方法, get方法中调用了 context = self.get_context_data() 然后 get 返回 context, 而 get_context_data() 方法的定义就是在 MultipleObjectTemplateResponseMixin 类中

五、关于Django原生自定义分页

默认情况下,使用 Django+Template 进行前后端结合在一起的开发,Django提供的Paginator类 已经能满足日常需求。

因为 Page 对象已经提供了关于分页操作的各种情况,这里做个汇总介绍

Paginator 属性和方法

属性

  • Paginator.count 返回的是所有页的对象总数

  • Paginator.num_pages 返回的是总的页数

  • Paginator.page_range 返回的是从1到最大页数的迭代器

方法

  • Paginator.get_page(page_number)

  • Paginator.Page(page_number)

以上两种方法都是用于获取当前页码对应的数据。不同的是:

1、返回值不一样

1.1、get_page 返回的是 Page 对象,可以通过该对象来访问页码、总页数、当前页的数据等信息。如下面对于 Page的属性和方法的汇总介绍

1.2、page 返回的是QuerySet对象, 可以像处理普通查询结果集一样处理分页数据。

2、处理异常情况不一样

如果指定的页码超出了有效范围(小于1或大于总页数)

2.1、get_page 方法会返回一个空的Page对象(不会抛出异常)

2.2、page 方法会抛出PageNotAnInteger或EmptyPage异常,需要在代码中进行处理。

Page 属性和方法

属性

  • Page.object_list 返回的是当前页码对应的对象列表

  • Page.number 返回的是当前页码

  • Page.paginator 放回的是对应的Paginator对象

方法

  • Page.has_next() 当前Page对象是否有下一页

  • Page.has_previous() 当前Page对象是否有上一页

  • Page.has_other_pages() 当前Page对象对应的页码是否有前一页或者后一页,如果有返回True

  • Page.next_page_number() 返回下一页的页码, 下一页不存在则报 InvaliPage

  • Page.previous_page_number() 返回上一页的页码, 上一页不存在则报 InvaliPage

  • Page.start_index() 返回当前页面的第一个对象 在所有对象中的索引值

  • Page.end_index() 返回当前页面的最后一个对象 在所有对象中的索引值


今天的知识就介绍到这里, 下一篇我们介绍在进行 Django前后端分离开发中,如何使用分页功能

如果觉得文章对你有用,请不吝点赞 和 关注个人公众号(搜索 全栈运维 或者 DailyJobOps)

附件 django_pagination示例代码下载

相关文章

  • 第20天,分页器

    本篇写了使用Django自有的分页器paginator的用法和自定制分页器 1.1 Django之分页功能 Dj...

  • Django常用-分页

    分页 Django提供了数据分页的类,这些类被定义在django/core/paginator.py中。 类Pag...

  • 分页显示

    说明: 分页  Django 提供了一些类实现管理数据分页,这些类位于 django/core/pagina...

  • Flask ----- 前端页面分页器对象

    Flask和Django都有的分页器类 Pageinations 分页器对象属性和方法 分页流程示例 借书管理系统...

  • 分页-Django REST框架

    分页-Django REST框架 pagination.py 分页 Django提供了几个类来帮助您管理分页数据-...

  • django 使用原生sql及分页

    python3 django 使用原生sql及分页 django 使用原生sql及分页,类似于django res...

  • django中自带分页类使用

    python---自定义分页类 1.基础使用: 后台数据获取: 前端代码: 2.扩展内置分页: 类修改: 类调用:...

  • 最浅显易懂的Django系列教程(33)-分页

    分页 在Django中实现分页功能非常简单。因为Django已经内置了两个处理分类的类。分别是Paginator和...

  • 安卓ListView

    ListView 如何提高其效率 复用ConvertView 自定义静态类ViewHolder 使用分页加载 使用...

  • 使用py3fdfs - 踩坑实录 __str__ return

    后台运营人员 通过django的admin页面,进行(图片)文件的上传 django使用 自定义的storage类...

网友评论

      本文标题:Django 自定义分页类和使用总结

      本文链接:https://www.haomeiwen.com/subject/ccukudtx.html