django 分页组件

一、仿django分页功能自己实现

urls.py


1

2

3

4

5

6

7

8

9

from django.conf.urls import url

from django.contrib import admin

from app01 import views

urlpatterns = [

    url(r‘^admin/‘, admin.site.urls),

    url(r‘^index.html$‘, views.index),

    ]

views.py


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

from django.shortcuts import render

from app01 import models

# Create your views here.

USER_LIST = []

#创建数据999条

for i in range(999):

    temp = {‘name‘:‘root‘+str(i),‘age‘:i }

    #加到userlist列表中

    USER_LIST.append(temp)

def index(request):

    #每页显示10条数据

    per_page_count = 10

    #current-page 当有页

    current_page = request.GET.get(‘p‘)

    #数字运算要转成int类型

    current_page = int(current_page)

    #如果是第1页,索引0-9,就是1-10的数

    #p=1

    #0,10   0-9  取索引

    #p=2

    #大于等于10,小于20就是10-19

    #10,20 10-19

    #start 开始页数  end=结束页数

    #如果p=1-1=0

    start = (current_page - 1) * per_page_count

    #1 * 10=10

    end = current_page*per_page_count

    #数据切片,每次显示10页

    data = USER_LIST[start:end]

    #上一页

    prev_pager = current_page -1

    #下一页

    next_pager = current_page +1

    return render(request,‘index.html‘,{‘user_list‘:data,‘prev_pager‘:prev_pager,‘next_pager‘:next_pager })

index.html


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

<!DOCTYPE html>

<html lang="en">

<head>

    <meta charset="UTF-8">

    <title>Title</title>

</head>

<body>

    <ul>

        {% for row in user_list %}

        <li>{{ row.name }}-{{ row.age }}</li>

        {% endfor %}

    </ul>

    <a href="/index.html?p={{ prev_pager }}">上一页</a>

    <a href="/index.html?p={{ next_pager }}">下一页</a>

</body>

</html>

二、利用django自带分页组件实现分页功能

使用分页器Paginator:
在视图中使用 Paginator来为查询集分页。我们提供视图以及相关的模板来展示如何展示这些结果。


1

2

3

4

5

Paginator常用属性

per_page: 每页显示条目数量

count: 数据总个数

num_pages:总页数

page_range:总页数的索引范围,页码的范围,从1开始,例如[1, 2, 3, 4]。

Paginator所需参数:


1

2

object_list 一个列表,元祖或则Django 的Queryset 对象 或则其他对象带有 count() or __len__()的方法

per_page :就是1页显示几条数据 

Paginator对象的方法:


1

page(number) :返回在提供的下标处的Page对象,下标以1开始。

使用page对象方法:


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

Page.has_next()

如果有下一页,则返回True。

Page.has_previous()

如果有上一页,返回 True。

Page.has_other_pages()

如果有上一页或下一页,返回True。

Page.next_page_number()

返回下一页的页码。如果下一页不存在,抛出InvalidPage异常。

Page.previous_page_number()

返回上一页的页码。如果上一页不存在,抛出InvalidPage异常。

Page.start_index()

返回当前页上的第一个对象,相对于分页列表的所有对象的序号,从1开始。比如,将五个对象的列表分为每页两个对象,第二页的start_index()会返回3。

Page.end_index()

返回当前页上的最后一个对象,相对于分页列表的所有对象的序号,从1开始。 比如,将五个对象的列表分为每页两个对象,第二页的end_index() 会返回 4。

属性


1

2

3

4

5

6

7

8

Page.object_list

当前页上所有对象的列表。

Page.number

当前页的序号,从1开始。

Page.paginator

相关的Paginator对象。

代码示例:

Django内置分页:Paginator、Page

urls.py


1

2

3

4

5

6

7

8

9

from django.conf.urls import url

from django.contrib import admin

from app01 import views

urlpatterns = [

    url(r‘^admin/‘, admin.site.urls),

    # url(r‘^index.html$‘, views.index),

    url(r‘^index1.html$‘, views.index1),

    ]

views.py


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

from django.shortcuts import render

from django.shortcuts import redirect

from django.shortcuts import HttpResponse

from app01 import models

# Create your views here.

USER_LIST = []

#创建数据999条

for i in range(999):

    temp = {‘name‘:‘root‘+str(i),‘age‘:i }

    #加到userlist列表中

    USER_LIST.append(temp)

def index1(request):

    from django.core.paginator import Paginator,EmptyPage,PageNotAnInteger

    #全部数据:USER_LIST,=>得出共有多少条数据

    # per_page: 每页显示条目数量

    # count:    数据总个数

    # num_pages:总页数

    # page_range:总页数的索引范围,如: (1,10),(1,200)

    # page:     page对象 (是否具有下一页,是否有上一页)

    current_page = request.GET.get(‘p‘)

    #Paginator对象,里面封装了上面那些值,把USER_LIST对象传过来了,显示10页

    paginator = Paginator(USER_LIST,10)

    try:

        #page对象

        #posts配置对象(current_page用户可能填些不合法的字段)

        #paginator通过拿到了page对象,把current_page传进来

        posts = paginator.page(current_page)

        # has_next              是否有下一页

        # next_page_number      下一页页码

        # has_previous          是否有上一页

        # previous_page_number  上一页页码

        # object_list           分页之后的数据列表,已经切片好的数据

        # number                当前页

        # paginator             paginator对象

   #表示你填的东西不是个整数

    except PageNotAnInteger:

        posts = paginator.page(1)

    #空页的时候,表示你看完了,显示最后一页

    except EmptyPage:

        posts = paginator.page(paginator.num_pages)

    return render(request,‘index1.html‘ ,{‘posts‘:posts})

index1.html


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

<!DOCTYPE html>

<html lang="en">

<head>

    <meta charset="UTF-8">

    <title>Title</title>

</head>

<body>

    <ul>

        {% for row in posts.object_list %}

            <li>{{ row.name }}-{{ row.age }}</li>

        {% endfor %}

    </ul>

   {% include ‘include/pager.html‘ %}

</body>

</html>

pager.html


1

2

3

4

5

6

7

8

9

10

11

12

13

14

{% if posts.has_previous %}

    <a href="/index1.html?p={{ posts.previous_page_number }}">上一页</a>

    {% else %}

    <a href="#">上一页</a>

{% endif %}

{% if posts.has_next %}

    <a href="/index1.html?p={{ posts.next_page_number }}">下一页</a>

{% endif %}

<span>

    {{ posts.number }} / {{ posts.paginator.num_pages }}

</span>

{#切片完后,就叫object_list#}

三、扩展Django内置分页

urls.py


1

2

3

4

5

6

7

from django.conf.urls import url

from django.contrib import admin

from  app01 import views

urlpatterns = [

    url(r‘^admin/‘, admin.site.urls),

    url(r‘^index/$‘, views.listing),

]

views.py


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

from django.shortcuts import render

from django.core.paginator import Paginator,EmptyPage,PageNotAnInteger

# Create your views here.

#模拟测试网页数据

USER_LIST = []

for i in range(1,999):

    temp = {"name":"root"+str(i),"age":i}

    USER_LIST.append(temp)

class CustomPaginator(Paginator):

    def __init__(self,current_page,per_pager_num,*args,**kwargs):

        # per_pager_num  显示的页码数量

        self.current_page = int(current_page)

        self.per_pager_num = int(per_pager_num)

        super(CustomPaginator,self).__init__(*args,**kwargs)

    def pager_num_range(self):

        ‘‘‘

        自定义显示页码数

        第一种:总页数小于显示的页码数

        第二种:总页数大于显示页数  根据当前页做判断  a 如果当前页大于显示页一半的时候  ,往右移一下

                                                b 如果当前页小于显示页的一半的时候,显示当前的页码数量

        第三种:当前页大于总页数

        :return:

        ‘‘‘

        if self.num_pages < self.per_pager_num:

            return range(1,self.num_pages+1)

        half_part = int(self.per_pager_num/2)

        if self.current_page <= half_part:

            return range(1,self.per_pager_num+1)

        if (self.current_page+half_part) > self.num_pages:

            return range(self.num_pages-self.per_pager_num+1,self.num_pages)

        return range(self.current_page-half_part,self.current_page+half_part+1)

def listing(request):

    current_page = request.GET.get(‘p‘)

    paginator = CustomPaginator(current_page,11,USER_LIST,10)

    try:

        paginator = paginator.page(current_page)  #获取前端传过来显示当前页的数据

    except PageNotAnInteger:

        # 如果有异常则显示第一页

        paginator = paginator.page(1)

    except EmptyPage:

        # 如果没有得到具体的分页内容的话,则显示最后一页

        paginator = paginator.page(paginator.num_pages)

    return render(request,‘index.html‘,{"users":paginator})

index.html


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

<!DOCTYPE html>

<html lang="en">

<head>

    <meta charset="UTF-8">

    <title>Title</title>

</head>

<body>

<ul>

    {% for user in users.object_list %}

        <li>{{ user.name }}-{{ user.age }}</li>

    {% endfor %}

    {% if users.has_previous %}

        <a href="/index?p={{ users.previous_page_number }}">上一页</a>

    {% endif %}

    {% for number in users.paginator.pager_num_range %}

        {% if number == users.number %}

            <a href="/index?p={{ number }}" style="font-size: 33px">{{ number }}</a>

        {% else %}

            <a href="/index?p={{ number }}" >{{ number }}</a>

        {% endif %}

    {% endfor %}

    {% if users.has_next %}

        <a href="/index?p={{ users.next_page_number }}">下一页</a>

    {% endif %}

    <span>{{ users.number }} /{{ users.paginator.num_pages }}</span>

</ul>

</body>

</html>

没加特效:

二、自定义分页器

效果:

分页功能在每个网站都是必要的,对于分页来说,其实就是根据用户的输入计算出应该在数据库表中的起始位置。


1

2

3

4

5

6

7

1、设定每页显示数据条数

2、用户输入页码(第一页、第二页...)

3、根据设定的每页显示条数和当前页码,计算出需要取数据表的起始位置

4、在数据表中根据起始位置取值,页面上输出数据

需求,需要在页面上显示分页的页面。如:[上一页][1][2][3][4][5][下一页]


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

1、设定每页显示数据条数

2、用户输入页码(第一页、第二页...)

3、设定显示多少页号

4、获取当前数据总条数

5、根据设定显示多少页号和数据总条数计算出,总页数

6、根据设定的每页显示条数和当前页码,计算出需要取数据表的起始位置

7、在数据表中根据起始位置取值,页面上输出数据

8、输出分页html,如:[上一页][1][2][3][4][5][下一页]

代码示例:

代码:

urls.py


1

2

3

4

5

6

7

8

from django.conf.urls import url

from django.contrib import admin

from app01 import views

urlpatterns = [

    url(r‘^admin/‘, admin.site.urls),

    url(r‘^index2.html$‘, views.index2),

]

views.py


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

from django.shortcuts import render

from django.core.paginator import Paginator, EmptyPage, PageNotAnInteger

#这里用的是手动生成数据,生产环境到数据库中去取数据就可以啦

USER_LIST = []

for i in range(1,666):

    temp = {‘name‘:‘root‘+str(i), ‘age‘:i}

    USER_LIST.append(temp)

def index2(request):

    from app01.pager import Pagination

    current_page = request.GET.get(‘p‘)

    page_obj = Pagination(666,current_page)

    data_list = USER_LIST[page_obj.start():page_obj.end()]

    return render(request,‘index2.html‘,{‘data‘:data_list,‘page_obj‘:page_obj})

pager.py


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

63

64

65

66

67

68

69

70

71

72

73

74

75

76

77

78

79

80

81

82

83

class Pagination(object):

    def __init__(self,totalCount,currentPage,perPageItemNum=10,maxPageNum=7):

        # 数据总个数

        self.total_count = totalCount

        # 当前页

        try:

            v = int(currentPage)

            if v <= 0:

               v = 1

            self.current_page = v

        except Exception as e:

            self.current_page = 1

        # 每页显示的行数

        self.per_page_item_num = perPageItemNum

        # 最多显示页面

        self.max_page_num = maxPageNum

    def start(self):

        return (self.current_page-1) * self.per_page_item_num

    def end(self):

        return self.current_page * self.per_page_item_num

    @property

    def num_pages(self):

        """

        总页数

        :return:

        """

        # 666

        # 10

        a,b = divmod(self.total_count,self.per_page_item_num)

        if b == 0:

            return a

        return a+1

    def pager_num_range(self):

        # self.num_pages()

        # self.num_pages

        # 当前页

        #self.current_page

        # 最多显示的页码数量 11

        #self.per_pager_num

        # 总页数

        # self.num_pages

        if self.num_pages < self.max_page_num:

            return range(1,self.num_pages+1)

        # 总页数特别多 5

        part = int(self.max_page_num/2)

        if self.current_page <= part:

            return range(1,self.max_page_num+1)

        if (self.current_page + part) > self.num_pages:

            return range(self.num_pages-self.max_page_num+1,self.num_pages+1)

        return range(self.current_page-part,self.current_page+part+1)

    def page_str(self):

        page_list = []

        first = "<li><a href=‘/index2.html?p=1‘>首页</a></li>"

        page_list.append(first)

        if self.current_page == 1:

            prev = "<li><a href=‘#‘>上一页</a></li>"

        else:

            prev = "<li><a href=‘/index2.html?p=%s‘>上一页</a></li>" %(self.current_page-1,)

        page_list.append(prev)

        for i in self.pager_num_range():

            if i == self.current_page:

                temp = "<li class=‘active‘><a href=‘/index2.html?p=%s‘>%s</a></li>" %(i,i)

            else:

                temp = "<li><a href=‘/index2.html?p=%s‘>%s</a></li>" % (i, i)

            page_list.append(temp)

        if self.current_page == self.num_pages:

            nex = "<li><a href=‘#‘>下一页</a></li>"

        else:

            nex = "<li><a href=‘/index2.html?p=%s‘>下一页</a></li>" % (self.current_page + 1,)

        page_list.append(nex)

        last = "<li><a href=‘/index2.html?p=%s‘>尾页</a></li>" %(self.num_pages,)

        page_list.append(last)

        return ‘‘.join(page_list)

index2.html


1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

<!DOCTYPE html>

<html lang="en">

<head>

    <meta charset="UTF-8">

    <title>Title</title>

    <link rel="stylesheet" href="/static/bootstrap-3.3.7-dist/css/bootstrap.css" />

</head>

<body>

    <ul>

        {% for row in data %}

            <li>{{ row.name }}-{{ row.age }}</li>

        {% endfor %}

    </ul>

    <ul class="pagination pagination-sm">

        {{ page_obj.page_str|safe }}

      </ul>

    <div style="height: 300px;"></div>

</body>

</html>

总结:分页时需要做三件事:

  • 创建处理分页数据的类
  • 根据分页数据获取数据
  • 输出分页HTML,即:[上一页][1][2][3][4][5][下一页]
时间: 2024-10-08 02:49:15

django 分页组件的相关文章

Django框架(十五)—— Django分页组件

Django分页组件 一.分页器 数据量大的话,可以分页获取,查看 例如:图书管理中,如果有成千上万本书,要是都在一个页面中渲染出来,会影响页面美观,所以就要用分页器分页渲染 二.分页器的使用 基本写法 基本写法: 后端: 总数据拿出来 生成分页器Paginator对象(对象里有属性和方法) 生成当前页的对象,current_page=paginator.page(当前页码) 取出前台传过来的页码,current_page_num = int(request.GET.get('page')) -

8.2 Django 分页组件替换自定义分页

Django的分页器(paginator) 总之不太好用我们还是用自己的好一些 自定义分页器 分页实现源码 """ 自定义分页组件 """ class Pagination(object): def __init__(self, current_page, all_count, base_url, params, per_page_num=8, pager_count=11, ): """ 封装分页相关数据 :par

Django框架(十三)--Django分页组件

一.分页器 数据量大的话,可以分页获取,查看 例如:图书管理中,如果有成千上万本书,要是都在一个页面中渲染出来,会影响页面美观,所以就要用分页器分页渲染 二.分页器的使用 基本写法 基本写法: 后端: 总数据拿出来 生成分页器Paginator对象(对象里有属性和方法) 生成当前页的对象,current_page=paginator.page(当前页码) 取出前台传过来的页码,current_page_num = int(request.GET.get('page')) -需要有异常捕获 -捕获

DJANGO 自定义分页组件

第一次在码云上面分享代码片段,记录一下: https://gitee.com/trunkslisa/codes/14gkxi3zf9e2ulbvjnqyo90 class Pagination(object): def __init__(self, totalCount, currentPage, perPageNum=20, maxPageNum=7): # 传入: # -所有数据的个数 totalCount self.totalCount = totalCount # -当前页 curren

Django的rest_framework的分页组件源码分析

前言: 分页大家应该都很清楚,今天我来给大家做一下Django的rest_framework的分页组件的分析:我的讲解的思路是这样的,分别使用APIview的视图类和基于ModelViewSet的视图类两种方式实现分页的功能,同时我也会介绍两个分页的类,PageNumberPagination类和LimitOffsetPagination,希望能对大家有所帮助! 今天的博客主要的这样的,先讲解基于APIView类的两种分页类的实现方式,然后在讲解基于ModelViewSet类的两种分页类的实现方

django -----分页器组件

1 Django的分页器(paginator)简介 2 应用View层 3 模版层 index.html 4 扩展 回到目录 1 Django的分页器(paginator)简介 在页面显示分页数据,需要用到Django分页器组件 from django.core.paginator import Paginator Paginator对象: paginator = Paginator(user_list, 10) # per_page: 每页显示条目数量 # count: 数据总个数 # num

DRF 分页组件

Django Rest Framework 分页组件 DRF的分页 为什么要使用分页 其实这个不说大家都知道,大家写项目的时候也是一定会用的, 我们数据库有几千万条数据,这些数据需要展示,我们不可能直接从数据库把数据全部读取出来, 这样会给内存造成特别大的压力,有可能还会内存溢出,所以我们希望一点一点的取, 那展示的时候也是一样的,总是要进行分页显示,我们之前自己都写过分页. 那么大家想一个问题,在数据量特别大的时候,我们的分页会越往后读取速度越慢, 当有一千万条数据,我要看最后一页的内容的时候

DjangoRestFramework学习三之认证组件、权限组件、频率组件、url注册器、响应器、分页组件

本节目录 一 认证组件 二 权限组件 三 频率组件 四 URL注册器 五 响应器 六 分页组件 七 xxx 八 xxx 一 认证组件 1. 局部认证组件 我们知道,我们不管路由怎么写的,对应的视图类怎么写的,都会走到dispatch方法,进行分发, 在咱们看的APIView类中的dispatch方法的源码中,有个self.initial(request, *args, **kwargs),那么认证.权限.频率这三个默认组件都在这个方法里面了,如果我们自己没有做这三个组件的配置,那么会使用源码中默

drf - 偏移分页组件 LimitOffsetPagination

目录 数据准备 分页组件部分源码分析一 自定义分页组件 分页组件部分源码分析二 数据准备 model.py文件 定义两个表Car表和Brand表,其中Car中的brand字段外键关联Brand表 from django.db import models class BaseModel(models.Model): is_delete = models.BooleanField(default=False) create_time = models.DateTimeField(auto_now_a