10. Django 分页器

Django 分页器

版权声明:本博客来自路飞学城Python全栈开发培训课件,仅用于学习之用,严禁用于商业用途。
欢迎访问路飞学城官网:https://www.luffycity.com/

本节重点

  • Django分页器相关类
  • Django分页器的实现
  • Django分页器的优化

1 Django分页器相关类

1.1 介绍

Django提供了一个内置类来帮助管理分页数据,这个模块存放在django.core.paginator.py。
其中有两个核心类,一个是Paginator类,另一个是Page类

1.2 Paginator类

初始化源码:

class Paginator(object):
    def __init__(self, object_list, per_page, orphans=0, allow_empty_first_page=True):
        self.object_list = object_list
        self.per_page = int(per_page)
        self.orphans = int(orphans)
        self.allow_empty_first_page = allow_empty_first_page
        self._num_pages = self._count = None

类方法:

  • Paginator.page(number):根据参数number返回一个Page对象

类属型:

  • Paginator.count:所有页面对象总数。
  • Pagnator.num_pages:页面总数。
  • pagiator.page_range:页面范围,从1开始,例如[1,2,3,4]。

1.3 Page类

初始化源码:

class Page(collections.Sequence):
    def __init__(self, object_list, number, paginator):
        self.object_list = object_list
        self.number = number
        self.paginator = paginator

类方法:

  • Page.has_next() : 如果有下一页,则返回True。
  • Page.has_previous() :如果有上一页,返回 True。
  • Page.has_other_pages() :如果有上一页或下一页,返回True。
  • Page.next_page_number() :返回下一页的页码。如果下一页不存在,抛出InvlidPage异常。
  • Page.previous_page_number() :返回上一页的页码。如果上一页不存在,抛出InvalidPage异常。
  • Page.start_index() :返回当前页上的第一个对象,相对于分页列表的所有对象的序号,从1开始。
    比如,将五个对象的列表分为每页两个对象,第二页的start_index()会返回3。
  • Page.end_index() :返回当前页上的最后一个对象,相对于分页列表的所有对象的序号,从1开始。
    比如,将五个对象的列表分为每页两个对象,第二页的end_index() 会返回 4。

类属型:

  • Page.object_list 当前页上所有对象的列表。
  • Page.number 当前页的序号,从1开始。
  • Page.paginator 相关的Paginator对象。

异常处理:

  • InvalidPage(Exception):异常的基类(后两个异常都是InalidPage的子类),当paginator传入一个无效的页码时抛出。 Paginator.page()返回在所请求的页面无效(比如不是一个整数)时,或者不包含任何对象时抛出异常。
  • exception PageNotAnInteger:当向page()提供一个不是整数的值时抛出。
  • exception EmptyPage:当向page()提供一个有效值,但是那个页面上没有任何对象时抛出。

2. Django分页器的实现

views.py文件代码:

# views.py
from django.shortcuts import render,HttpResponse
from django.core.paginator import Paginator, EmptyPage, PageNotAnInteger

from app01.models import *


def index(request):
    # 迭代all()时django会自动分块加载数据,并不会一次性加载所有数据到内存,可以放心遍历all()
    # 但是仍然要注意大数据内存的使用,比如list(objects.all())此类操作
    book_list=Book.objects.all() # 获取所有的book
    paginator = Paginator(book_list, 10) # 每页10条
    page = request.GET.get('page',1) # 获取页面请求的page页码,默认为第1页
    currentPage=int(page)

    try:
        print(page)
        book_list = paginator.page(page) # book_list为page对象
    except PageNotAnInteger:
        # If page is not an integer, deliver first page.
        book_list = paginator.page(1) #
    except EmptyPage:
        # If page is out of range (e.g. 9999), deliver last page of results.
        book_list = paginator.page(paginator.num_pages)

    return render(request,"index.html",
                  {"book_list":book_list,"paginator":paginator,"currentPage":currentPage})

HTML文件代码:

# index.html
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <link rel="stylesheet" href="https://cdn.bootcss.com/bootstrap/3.3.7/css/bootstrap.min.css" 
    integrity="sha384-BVYiiSIFeK1dGmJRAkycuHAHRg32OmUcww7on3RYdg4Va+PmSTsz/K68vbdEjh4u" crossorigin="anonymous">
</head>
<body>

<div class="container">

    <h4>分页器</h4>
    <ul>
        {% for book in book_list %}
             <li>{{ book.title }} -----{{ book.price }}</li>
        {% endfor %}
     </ul>
    <ul class="pagination" id="pager">
        {% if book_list.has_previous %}
           <li class="previous">
   			<a href="/index/?page={{ book_list.previous_page_number }}">上一页</a>
   		</li>
        {% else %}
           <li class="previous disabled"><a href="#">上一页</a></li>
        {% endif %}

        {% for num in paginator.page_range %}
            {% if num == currentPage %}
              <li class="item active"><a href="/index/?page={{ num }}">{{ num }}</a></li>
            {% else %}
              <li class="item"><a href="/index/?page={{ num }}">{{ num }}</a></li>
            {% endif %}                        
        {% endfor %}

        {% if book_list.has_next %}
           <li class="next">
           	<a href="/index/?page={{ book_list.next_page_number }}">下一页</a>
           </li>
        {% else %}
           <li class="next disabled"><a href="#">下一页</a></li>
        {% endif %}

    </ul>
</div>
</body>
</html>

以上,便实现了简单版的分页显示效果,若我们数据量很大时,会出现如下效果:

在这里插入图片描述

很显然,让所有的页签数都显示出来,不太合理,需要优化下。

3. Django分页器的优化

当页数实在太多时,我们不可能把所有的页数全部显示出来,比如固定显示7页,当点击“中间”一些页数的时候,两边固定显示前3页和后3页,改动如下:

views.py文件代码:

# views.py
from django.shortcuts import render,HttpResponse
from django.core.paginator import Paginator, EmptyPage, PageNotAnInteger

from app01.models import *


def index(request):
    book_list=Book.objects.all()
    paginator = Paginator(book_list, 3)
    page = request.GET.get('page',1)
    currentPage=int(page)
    
    if paginator.num_pages>7:
		#  如果大于7,分情况处理
        if currentPage-3<1:
            pageRange=range(1,8)
        elif currentPage+3>paginator.num_pages:
            pageRange=range(currentPage-6,paginator.num_pages+1) #左开右闭
        else:
            pageRange=range(currentPage-3,currentPage+4)
    else:
        pageRange=paginator.page_range

    try:
        print(page)
        book_list = paginator.page(page)
    except PageNotAnInteger:
        book_list = paginator.page(1)
    except EmptyPage:
        book_list = paginator.page(paginator.num_pages)

    return render(request,"index.html",locals())

HTML文件代码修改如下:


		...
		
        {% for num in pageRange %}
            {% if num == currentPage %}
              <li class="item active"><a href="/index/?page={{ num }}">{{ num }}</a></li>
            {% else %}
              <li class="item"><a href="/index/?page={{ num }}">{{ num }}</a></li>
            {% endif %}                        
        {% endfor %}
        
		...

显示效果:

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
Django中,分页是一个常见的需求,为了方便处理分页,可以使用Django内置的分页Paginator。 Paginator的使用方法很简单,我们可以将一个QuerySet对象或者一个列表作为参数传入Paginator中,然后就可以使用Paginator提供的一些方法来进行分页操作。下面是一个简单的分页小插件的代码: ```python from django.core.paginator import Paginator, EmptyPage, PageNotAnInteger def paginate(objects, page=1, per_page=10): """ 分页小插件 """ paginator = Paginator(objects, per_page) try: paginated_objects = paginator.page(page) except PageNotAnInteger: paginated_objects = paginator.page(1) except EmptyPage: paginated_objects = paginator.page(paginator.num_pages) return paginated_objects ``` 上面的代码定义了一个名为paginate的函数,它接受三个参数: - objects:需要进行分页的对象,可以是一个QuerySet对象或者一个列表。 - page:当前页码,默认为1。 - per_page:每页显示的条目数量,默认为10。 函数中首先使用Paginator来对objects进行分页操作,并使用try/except语句来处理异常情况。如果page参数不是一个整数,则将page设置为1;如果page参数超出了可用的页码范围,则将page设置为最大页码数(paginator.num_pages)。 最后,函数返回一个分页后的对象(paginated_objects),可以在Django模板中使用该对象来渲染分页。 使用该分页小插件的示例代码如下: ```python from django.shortcuts import render from .models import MyModel from .utils import paginate def my_view(request): my_objects = MyModel.objects.all() paginated_objects = paginate(my_objects, request.GET.get('page')) return render(request, 'my_template.html', {'paginated_objects': paginated_objects}) ``` 在上面的代码中,我们首先获取所有的MyModel对象,然后使用paginate函数将其分页,最后将分页后的对象传递给模板进行渲染。 在模板中,可以使用Django内置的分页标签({% for obj in paginated_objects %}、{% if paginated_objects.has_previous %}等)来渲染分页

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值