Django:自定义分页
相关知识点
QueryDict类型
1、用户发送get请求,其信息是存储在 request.GET 中,而request.GET 是QueryDict类型的,该类型取值的方式与普通的字典类似,但是该字典是可读,不可写的,如果想要修改该字典中的内容需要对其进行深拷贝,才能修改
{"user":user,"page":2} # QueryDict类型,与普通字典长的一样,不可修改
2、如何修改?(深拷贝后得到的可以修改)
# 若请求中 ?a=1&b=2 request.GET["xxx"]=123 print(type(request.GET)) # 报错 不可直接修改 import copy params = copy.deepcopy(request.GET) #深拷贝后可以修改 params["xxx"]=123 print("params",params) print(params.urlencode()) # "a=1&b=2&xxx=123"
分析:通过 urlencode()方法可以反解为请求时的 urlencode数据类型,因此该方法可以用于保存搜索条件,通过将新的参数添加到 parmars 中,然后转成 urlencode类型就可以作为,请求时的路径,此时路径就保存了搜索路径的信息。
3、请求的分类
post 请求:一般是用来提交数据的(添加和修改),参数会有 contentType 数据格式
get 请求:一般是用来查询的,参数是固定的 urlencode 类型
delete 请求:一般是用来删除的,
自定义分页
需求
数据库中有大量的数据时,在分多页展示的时候,需要用到分页展示数据,
首先想数据库中批量插入数据:
#批量插入数据: # for i in range(100): # Book.objects.create(title="book_%s"%i,price=i*i) #该方法较慢 book_list=[] for i in range(100): book=Book(title="book_%s"%i,price=i*i) book_list.append(book) Book.objects.bulk_create(book_list)
具体实现代码:
该方法实现了 保存搜索条件 的功能
1、分页的使用(views.py)
from app01.models import Book from django.core.paginator import Paginator,EmptyPage def index(request):from app01.page import Pagination # 引入分页的类 current_page_num = request.GET.get("page",1) # 从发送的请求中,?参数,获取值 book_list = Book.objects.all()
pagination=Pagination(current_page_num,book_list.count(),request) # 当前页码、总页数、request book_list=book_list[pagination.start:pagination.end] # pagination.start可以直接得到开始页的值 return render(request,"index.html",locals())
2、创建 Pagination 类用于分页展示(app01中创建 page.py 用来成名分页的类)
class Pagination(object): def __init__(self,current_page_num,all_count,request,per_page_num=10,pager_count=11,): #默认参数可以改变 """ 封装分页相关数据 :param current_page_num: 当前访问页的数字 :param all_count: 分页数据中的数据总条数
:request:里面request.GET中含有get请求发送的键值对
:param per_page_num: 每页显示的数据条数 :param pager_count: 最多显示的页码个数 """ try: current_page_num = int(current_page_num) except Exception as e: current_page_num = 1 if current_page_num <1: current_page_num = 1 self.current_page_num = current_page_num self.all_count = all_count self.per_page_num = per_page_num # 实际总页码 all_pager, tmp = divmod(all_count, per_page_num) if tmp: all_pager += 1 self.all_pager = all_pager self.pager_count = pager_count self.pager_count_half = int((pager_count - 1) / 2) # 5 # 保存搜索条件 import copy self.params=copy.deepcopy(request.GET) # {"a":"1","b":"2"} @property def start(self): return (self.current_page_num - 1) * self.per_page_num @property def end(self): return self.current_page_num * self.per_page_num # 该方法用于在页面渲染分页的标签按钮 def page_html(self): # 如果总页码 < 11个: if self.all_pager <= self.pager_count: pager_start = 1 pager_end = self.all_pager + 1 # 总页码 > 11 else: # 当前页如果<=页面上最多显示11/2个页码 if self.current_page_num <= self.pager_count_half: pager_start = 1 pager_end = self.pager_count + 1 # 当前页大于5 else: # 页码翻到最后 if (self.current_page_num + self.pager_count_half) > self.all_pager: pager_start = self.all_pager - self.pager_count + 1 pager_end = self.all_pager + 1 else: pager_start = self.current_page_num - self.pager_count_half pager_end = self.current_page_num + self.pager_count_half + 1 page_html_list = [] # 存放分页标签,用于整体渲染 # 首页 self.params["page"]=1 # 添加键值对,page=1 first_page = '<li><a href="?%s">首页</a></li>' % (self.params.urlencode(),) page_html_list.append(first_page) # 上一页 self.params["page"]=self.current_page_num - 1 if self.current_page_num <= 1: prev_page = '<li class="disabled"><a href="#">上一页</a></li>' else: prev_page = '<li><a href="?%s">上一页</a></li>' % (self.params.urlencode(),) page_html_list.append(prev_page) # 中间页 for i in range(pager_start, pager_end): self.params["page"]=i if i == self.current_page_num: temp = '<li class="active"><a href="?%s">%s</a></li>' %(self.params.urlencode(),i) else: temp = '<li><a href="?%s">%s</a></li>' % (self.params.urlencode(),i,) # 保存搜索条件 page_html_list.append(temp) # 下一页 self.params["page"]=self.current_page_num + 1 if self.current_page_num >= self.all_pager: next_page = '<li class="disabled"><a href="#">下一页</a></li>' else: next_page = '<li><a href="?%s">下一页</a></li>' % (self.params.urlencode(),) page_html_list.append(next_page) # 尾页 self.params["page"]=self.all_pager last_page = '<li><a href="?%s">尾页</a></li>' % (self.params.urlencode(),) page_html_list.append(last_page) return ''.join(page_html_list)
3、页面的渲染(html)
借助bootstrap的样式
<!DOCTYPE html> <html lang="zh-CN"> <head> <meta charset="UTF-8"> <title>Title</title> <meta name="viewport" content="width=device-width, initial-scale=1"> <!-- 最新版本的 Bootstrap 核心 CSS 文件 --> <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@3.3.7/dist/css/bootstrap.min.css" integrity="sha384-BVYiiSIFeK1dGmJRAkycuHAHRg32OmUcww7on3RYdg4Va+PmSTsz/K68vbdEjh4u" crossorigin="anonymous"> </head> <body> <!-- 渲染数据库的内容 --> <ul> {% for book in book_list %} <li>{{ book.title }} ---- {{ book.price }}</li> {% endfor %} </ul> <!-- 生成分页的标签 借助bootstrap的样式 nav ul--> <nav aria-label="Page navigation"> <ul class="pagination"> {{ pagination.page_html|safe }} </ul> </nav> </body> </html>
- 自定义分页参考链接:
http://www.cnblogs.com/haiyan123/p/8065353.html 该文中与本文中Pagination 中传递的参数不一样,功能一样,本文中只有中间页可以保存搜索路径,需要进一步修改。

浙公网安备 33010602011771号