Django---分页器、中间件

朱雀 2021-09-17 15:50 424阅读 0赞

分页

Django的分页器(paginator)

view

  1. from django.shortcuts import render,HttpResponse
  2. # Create your views here.
  3. from app01.models import *
  4. from django.core.paginator import Paginator, EmptyPage, PageNotAnInteger
  5. def index(request):
  6. '''
  7. 批量导入数据:
  8. Booklist=[]
  9. for i in range(100):
  10. Booklist.append(Book(title="book"+str(i),price=30+i*i))
  11. Book.objects.bulk_create(Booklist)
  12. '''
  13. '''
  14. 分页器的使用:
  15. book_list=Book.objects.all()
  16. paginator = Paginator(book_list, 10)
  17. print("count:",paginator.count) #数据总数
  18. print("num_pages",paginator.num_pages) #总页数
  19. print("page_range",paginator.page_range) #页码的列表
  20. page1=paginator.page(1) #第1页的page对象
  21. for i in page1: #遍历第1页的所有数据对象
  22. print(i)
  23. print(page1.object_list) #第1页的所有数据
  24. page2=paginator.page(2)
  25. print(page2.has_next()) #是否有下一页
  26. print(page2.next_page_number()) #下一页的页码
  27. print(page2.has_previous()) #是否有上一页
  28. print(page2.previous_page_number()) #上一页的页码
  29. # 抛错
  30. #page=paginator.page(12) # error:EmptyPage
  31. #page=paginator.page("z") # error:PageNotAnInteger
  32. '''
  33. book_list=Book.objects.all()
  34. paginator = Paginator(book_list, 10)
  35. page = request.GET.get('page',1)
  36. currentPage=int(page)
  37. try:
  38. print(page)
  39. book_list = paginator.page(page)
  40. except PageNotAnInteger:
  41. book_list = paginator.page(1)
  42. except EmptyPage:
  43. book_list = paginator.page(paginator.num_pages)
  44. return render(request,"index.html",{
  45. "book_list":book_list,"paginator":paginator,"currentPage":currentPage})

index.html:

  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4. <meta charset="UTF-8">
  5. <title>Title</title>
  6. <link rel="stylesheet" href="https://cdn.bootcss.com/bootstrap/3.3.7/css/bootstrap.min.css"
  7. integrity="sha384-BVYiiSIFeK1dGmJRAkycuHAHRg32OmUcww7on3RYdg4Va+PmSTsz/K68vbdEjh4u" crossorigin="anonymous">
  8. </head>
  9. <body>
  10. <div class="container">
  11. <h4>分页器</h4>
  12. <ul>
  13. {% for book in book_list %}
  14. <li>{
  15. { book.title }} -----{
  16. { book.price }}</li>
  17. {% endfor %}
  18. </ul>
  19. <ul class="pagination" id="pager">
  20. {% if book_list.has_previous %}
  21. <li class="previous"><a href="/index/?page={
  22. { book_list.previous_page_number }}">上一页</a></li>
  23. {% else %}
  24. <li class="previous disabled"><a href="#">上一页</a></li>
  25. {% endif %}
  26. {% for num in paginator.page_range %}
  27. {% if num == currentPage %}
  28. <li class="item active"><a href="/index/?page={
  29. { num }}">{
  30. { num }}</a></li>
  31. {% else %}
  32. <li class="item"><a href="/index/?page={
  33. { num }}">{
  34. { num }}</a></li>
  35. {% endif %}
  36. {% endfor %}
  37. {% if book_list.has_next %}
  38. <li class="next"><a href="/index/?page={
  39. { book_list.next_page_number }}">下一页</a></li>
  40. {% else %}
  41. <li class="next disabled"><a href="#">下一页</a></li>
  42. {% endif %}
  43. </ul>
  44. </div>
  45. </body>
  46. </html>

扩展

  1. def index(request):
  2. book_list=Book.objects.all()
  3. paginator = Paginator(book_list, 15)
  4. page = request.GET.get('page',1)
  5. currentPage=int(page)
  6. # 如果页数十分多时,换另外一种显示方式
  7. if paginator.num_pages>30:
  8. if currentPage-5<1:
  9. pageRange=range(1,11)
  10. elif currentPage+5>paginator.num_pages:
  11. pageRange=range(currentPage-5,paginator.num_pages+1)
  12. else:
  13. pageRange=range(currentPage-5,currentPage+5)
  14. else:
  15. pageRange=paginator.page_range
  16. try:
  17. print(page)
  18. book_list = paginator.page(page)
  19. except PageNotAnInteger:
  20. book_list = paginator.page(1)
  21. except EmptyPage:
  22. book_list = paginator.page(paginator.num_pages)
  23. return render(request,"index.html",locals())

中间件

101822420536468.png

中间件的概念

中间件顾名思义,是介于request与response处理之间的一道处理过程,相对比较轻量级,并且在全局上改变django的输入与输出。因为改变的是全局,所以需要谨慎实用,用不好会影响到性能。

Django的中间件的定义:

?










1



Middleware
is
a framework of hooks into Django’s request
/
response processing. <br>It’s a light, low
-
level “plugin” system
for
globally altering Django’s
input
or
output.

如果你想修改请求,例如被传送到view中的HttpRequest对象。 或者你想修改view返回的HttpResponse对象,这些都可以通过中间件来实现。

可能你还想在view执行之前做一些操作,这种情况就可以用 middleware来实现。

大家可能频繁在view使用request.user吧。 Django想在每个view执行之前把user设置为request的属性,于是就用了一个中间件来实现这个目标。所以Django提供了可以修改request 对象的中间件 AuthenticationMiddleware

Django默认的Middleware

  1. MIDDLEWARE = [
  2. 'django.middleware.security.SecurityMiddleware',
  3. 'django.contrib.sessions.middleware.SessionMiddleware',
  4. 'django.middleware.common.CommonMiddleware',
  5. 'django.middleware.csrf.CsrfViewMiddleware',
  6. 'django.contrib.auth.middleware.AuthenticationMiddleware',
  7. 'django.contrib.messages.middleware.MessageMiddleware',
  8. 'django.middleware.clickjacking.XFrameOptionsMiddleware',
  9. ]

每一个中间件都有具体的功能。

自定义中间件

中间件中一共有四个方法:

  1. process_request
  2. process_view
  3. process_exception
  4. process_response

process_request,process_response

当用户发起请求的时候会依次经过所有的的中间件,这个时候的请求时process_request,最后到达views的函数中,views函数处理后,在依次穿过中间件,这个时候是process_response,最后返回给请求者。

877318-20171012212952512-1143032176.png

上述截图中的中间件都是django中的,我们也可以自己定义一个中间件,我们可以自己写一个类,但是必须继承MiddlewareMixin

需要导入

?










1



from
django.utils.deprecation
import
MiddlewareMixin

877318-20171012215322324-2079210800.png

注意:在使用之前需要在settings中配置一下,在MIDDLEWARE中加上我们自定义的中间件,注意顺序,从上往下加载读取,这个顺序会影响执行顺序

in views:

  1. def index(request):
  2. print("view函数...")
  3. return HttpResponse("OK")

in Mymiddlewares.py:

  1. from django.utils.deprecation import MiddlewareMixin
  2. from django.shortcuts import HttpResponse
  3. class Md1(MiddlewareMixin):
  4. def process_request(self,request):
  5. print("Md1请求")
  6. def process_response(self,request,response):
  7. print("Md1返回")
  8. return response
  9. class Md2(MiddlewareMixin):
  10. def process_request(self,request):
  11. print("Md2请求")
  12. #return HttpResponse("Md2中断")
  13. def process_response(self,request,response):
  14. print("Md2返回")
  15. return response

结果:

  1. Md1请求
  2. Md2请求
  3. view函数...
  4. Md2返回
  5. Md1返回

注意:如果当请求到达请求2的时候直接不符合条件返回,即return HttpResponse(“Md2中断”),程序将把请求直接发给中间件2返回,然后依次返回到请求者,结果如下:

返回Md2中断的页面,后台打印如下:

  1. Md1请求
  2. Md2请求
  3. Md2返回
  4. Md1返回

流程图如下:

877318-20171012223730527-1366794845.png

即:

当 def process_request

# 无返回值:继续执行后续中间件和视图函数
# 有返回值:执行自己的process_response和上面的response

process_view

?










1



process_view(
self
, request, callback, callback_args, callback_kwargs)

Mymiddlewares.py修改如下

  1. from django.utils.deprecation import MiddlewareMixin
  2. from django.shortcuts import HttpResponse
  3. class Md1(MiddlewareMixin):
  4. def process_request(self,request):
  5. print("Md1请求")
  6. #return HttpResponse("Md1中断")
  7. def process_response(self,request,response):
  8. print("Md1返回")
  9. return response
  10. def process_view(self, request, callback, callback_args, callback_kwargs):
  11. print("Md1view")
  12. class Md2(MiddlewareMixin):
  13. def process_request(self,request):
  14. print("Md2请求")
  15. return HttpResponse("Md2中断")
  16. def process_response(self,request,response):
  17. print("Md2返回")
  18. return response
  19. def process_view(self, request, callback, callback_args, callback_kwargs):
  20. print("Md2view")

结果如下:

  1. Md1请求
  2. Md2请求
  3. Md1view
  4. Md2view
  5. view函数...
  6. Md2返回
  7. Md1返回

下图进行分析上面的过程:

997599-20170113093429385-1950865037.png

当最后一个中间的process_request到达路由关系映射之后,返回到中间件1的process_view,然后依次往下,到达views函数,最后通过process_response依次返回到达用户。

process_view可以用来调用视图函数:

  1. class Md1(MiddlewareMixin):
  2. def process_request(self,request):
  3. print("Md1请求")
  4. #return HttpResponse("Md1中断")
  5. def process_response(self,request,response):
  6. print("Md1返回")
  7. return response
  8. def process_view(self, request, callback, callback_args, callback_kwargs):
  9. # return HttpResponse("hello")
  10. response=callback(request,*callback_args,**callback_kwargs)
  11. return response

结果如下:

  1. Md1请求
  2. Md2请求
  3. view函数...
  4. Md2返回
  5. Md1返回

注意:process_view如果有返回值,会越过其他的process_view以及视图函数,但是所有的process_response都还会执行。

process_exception

?










1



process_exception(
self
, request, exception)

示例修改如下:

  1. class Md1(MiddlewareMixin):
  2. def process_request(self,request):
  3. print("Md1请求")
  4. #return HttpResponse("Md1中断")
  5. def process_response(self,request,response):
  6. print("Md1返回")
  7. return response
  8. def process_view(self, request, callback, callback_args, callback_kwargs):
  9. # return HttpResponse("hello")
  10. # response=callback(request,*callback_args,**callback_kwargs)
  11. # return response
  12. print("md1 process_view...")
  13. def process_exception(self):
  14. print("md1 process_exception...")
  15. class Md2(MiddlewareMixin):
  16. def process_request(self,request):
  17. print("Md2请求")
  18. # return HttpResponse("Md2中断")
  19. def process_response(self,request,response):
  20. print("Md2返回")
  21. return response
  22. def process_view(self, request, callback, callback_args, callback_kwargs):
  23. print("md2 process_view...")
  24. def process_exception(self):
  25. print("md1 process_exception...")

结果如下:

  1. Md1请求
  2. Md2请求
  3. md1 process_view...
  4. md2 process_view...
  5. view函数...
  6. Md2返回
  7. Md1返回

流程图如下:

当views出现错误时:

877318-20171012235627512-66918024.png

将md2的process_exception修改如下:

  1. def process_exception(self,request,exception):
  2. print("md2 process_exception...")
  3. return HttpResponse("error")

结果如下:

  1. Md1请求
  2. Md2请求
  3. md1 process_view...
  4. md2 process_view...
  5. view函数...
  6. md2 process_exception...
  7. Md2返回
  8. Md1返回

发表评论

表情:
评论列表 (有 0 条评论,424人围观)

还没有评论,来说两句吧...

相关阅读

    相关 django

    分页 Paginator 分页器 Paginator用于分页,但Paginator并不具体管理具体的页的处理,而是使用Page对象管理具体页面 创建分页器对

    相关 django中间

    中间件函数是django框架给我们预留的函数接口,让我们可以干预请求和应答的过程。 ![1682918-20190610110717958-1367766905.png][]