一、用户认证组件

1、auth模块

  1. from django.conrtrib import auth

django.contrib.auth中提供了许多方法,这里主要介绍其中的三个:

1)authenticate()

  它提供了用户认证,即验证用户名以及密码是否正确,一般需要username和password两个关键字参数。如果认证信息有效,会返回一个 User 对象。authenticate()会在User 对象上设置一个属性标识那种认证后端认证了该用户,且该信息在后面的登录过程中是需要的。当我们试图登录一个从数据库中直接取出来不经过authenticate()的User对象会报错的!!

  1. user_obj = auth.authenticate(username="someone",password="somepassword")
  2.   # 去auth_user表中查询记录,查询成功则返回用户对象,查询失败返回None

2)login(HttpRequest, user)

   该函数接受一个HttpRequest对象,以及一个认证了的User对象。

此函数使用django的session框架给某个已认证的用户附加上sessionid等信息。

  1. from django.contrib import auth
  2.  
  3.   def my_view(request):
  4.     username = request.POST['username']
  5.     password = request.POST['password']
  6.     user_obj = auth.authenticate(username=username, password=password)
  7.     if not user_obj: # 成立表示去auth_user表认证成功
  8.       auth.login(request, user_obj) # 则保存用户状态
  9.       # Redirect to a success page.
  10.     else:
  11.       # Return an 'invalid login' error message.

3)logout(HttpRequest) - 注销用户  

该函数接受一个HttpRequest对象,无返回值。当调用该函数时,当前请求的session信息会全部清除。该用户即使没有登录,使用该函数也不会报错。

  1. from django.contrib import auth
  2.  
  3.   def logout_view(request):
  4.     auth.logout(request)
  5.     # Redirect to a logout success page

2、User对象

  1. from django.contrib.auth.models import User

User 对象属性:username,password(这两个是必填项,password用哈希算法保存到数据库)

1) User对象的is_authenticated()

  如果是真正的 User 对象,is_authenticated()的返回值恒为 True 。用于检查用户是否已经通过了认证。通过认证并不意味着用户拥有任何权限,甚至也不检查该用户是否处于激活状态,这只是表明用户成功的通过了认证。这个方法很重要, 在后台用request.user.is_authenticated()判断用户是否已经登录,如果true则可以向前台展示request.user.username。

现有需求:

a、用户登陆后才能访问某些页面;

b、如果用户没有登录就访问该页面的话直接跳到登录页面;

c、用户在跳转的登陆界面中完成登陆后,自动访问跳转到之前访问的地址;

方法一:

  1. def my_view(request):
  2.     if not request.user.is_authenticated():
  3.       return redirect('%s?next=%s' % (settings.LOGIN_URL, request.path))

方法二:django已经为我们设计好了一个用于此种情况的装饰器:login_requierd()

  1. from django.contrib.auth.decorators import login_required
  2.   @login_required
  3.   def my_view(request):
  4.     ...

  若用户没有登录,则会跳转到django默认的登录URL '/accounts/login/ ' (这个值可以在settings文件中通过LOGIN_URL进行修改,如:LOGIN_URL='/login/')。并传递当前访问url的绝对路径 (登陆成功后,会重定向到该路径)。

2)创建用户(使用create_user辅助函数创建用户)

  1.   from django.contrib.auth.models import User
  2.  
  3.   # 向auth_user表中添加一条用户信息记录
  4.   user = User.objects.create_userusername='',password='',email=''
  5.   # 跟以前的objects.create()的区别是create_user添加的记录密码字段是加密的

3)密码检查

  1.  # 用户要修改密码的时候,首先让他输入原来的密码,若给定的字符串通过了密码检查,返回True
  2.   check_password(passwd)

4)修改密码

  1. # 使用set_password() 来修改密码
  2.   user = User.objects.get(username=request.user.username)
  3.   user.set_password(raw_password="newpassword")
  4.   user.save()

3、简单示例

注册:

  1. def sign_up(request):
  2.     state = None
  3.     if request.method == 'POST':
  4.       password = request.POST.get('password', '')
  5.       repeat_password = request.POST.get('repeat_password', '')
  6.       email = request.POST.get('email', '')
  7.       username = request.POST.get('username', '')
  8.       if User.objects.filter(username=username):
  9.         state = 'user_exist'
  10.       else:
  11.         new_user=User.objects.create_user(username=username, password=password,email=email)
  12.         new_user.save()
  13.  
  14.         return redirect('/book/')
  15.     content = {
  16.       'state': state,
  17.       'user': None,}
  18.     return render(request, 'sign_up.html', content)

修改密码:

  1. @login_required
  2.   def set_password(request):
  3.     user = request.user
  4.     state = None
  5.     if request.method == 'POST':
  6.       old_password = request.POST.get('old_password', '')
  7.       new_password = request.POST.get('new_password', '')
  8.       repeat_password = request.POST.get('repeat_password', '')
  9.       if user.check_password(old_password):
  10.         if not new_password:
  11.           state = 'empty'
  12.         elif new_password != repeat_password:
  13.           state = 'repeat_error'
  14.         else:
  15.           user.set_password(new_password)
  16.           user.save()
  17.           return redirect("/log_in/")
  18.       else:
  19.         state = 'password_error'
  20.     content = {
  21.       'user': user,
  22.       'state': state,
  23.     }
  24.     return render(request, 'set_password.html', content)

二、QueryDict类型

1、用户发送get请求,其信息是存储在 request.GET 中,而request.GET 是QueryDict类型的,该类型取值的方式与普通的字典类似,但是该字典是可读,不可写的,如果想要修改该字典中的内容需要对其进行深拷贝,才能修改

  1. {"user":user,"page":2} # QueryDict类型,与普通字典长的一样,不可修改

2、如何修改?(深拷贝后得到的可以修改)

  1. # 若请求中 ?a=1&b=2
  2.  
  3. request.GET["xxx"]=123
  4. print(type(request.GET)) # 报错 不可直接修改
  5.  
  6. import copy
  7. params = copy.deepcopy(request.GET) #深拷贝后可以修改
  8. params["xxx"]=123
  9. print("params",params)
  10.  
  11. print(params.urlencode()) # "a=1&b=2&xxx=123"

分析:通过 urlencode()方法可以反解为请求时的 urlencode数据类型,因此该方法可以用于保存搜索条件,通过将新的参数添加到 parmars 中,然后转成 urlencode类型就可以作为,请求时的路径,此时路径就保存了搜索路径的信息。

3、请求的分类

post 请求:一般是用来提交数据的(添加和修改),参数会有 contentType 数据格式

get 请求:一般是用来查询的,参数是固定的 urlencode 类型

delete 请求:一般是用来删除的,

三、自定义分页

需求

数据库中有大量的数据时,在分多页展示的时候,需要用到分页展示数据,

首先想数据库中批量插入数据:

  1. #批量插入数据:
  2. # for i in range(100):
  3. # Book.objects.create(title="book_%s"%i,price=i*i) #该方法较慢
  4.  
  5. book_list=[]
  6. for i in range(100):
  7. book=Book(title="book_%s"%i,price=i*i)
  8. book_list.append(book)
  9.  
  10. Book.objects.bulk_create(book_list)

具体实现代码:

  该方法实现了 保存搜索条件 的功能

1、分页的使用(views.py)

  1. from app01.models import Book
  2. from django.core.paginator import Paginator,EmptyPage
  3. def index(request):from app01.page import Pagination # 引入分页的类
  4.  
  5. current_page_num = request.GET.get("page"1) # 从发送的请求中,?参数,获取值
  6. book_list = Book.objects.all()
  7.  
  8. pagination=Pagination(current_page_num,book_list.count(),request) # 当前页码、总页数、request
  9.   
  10. book_list=book_list[pagination.start:pagination.end] # pagination.start可以直接得到开始页的值
  11.  
  12. return render(request,"index.html",locals())

2、创建 Pagination 类用于分页展示(app01中创建 page.py 用来成名分页的类)

  1. class Pagination(object):
  2.  
  3. def __init__(self,current_page_num,all_count,request,per_page_num=10,pager_count=11,): #默认参数可以改变
  4. """
  5. 封装分页相关数据
  6. :param current_page_num: 当前访问页的数字
  7. :param all_count: 分页数据中的数据总条数
  8.      :request:里面request.GET中含有get请求发送的键值对
  9.  
  10. :param per_page_num: 每页显示的数据条数
  11. :param pager_count: 最多显示的页码个数
  12. """
  13. try:
  14. current_page_num = int(current_page_num)
  15. except Exception as e:
  16. current_page_num = 1
  17.  
  18. if current_page_num <1:
  19. current_page_num = 1
  20.  
  21. self.current_page_num = current_page_num
  22.  
  23. self.all_count = all_count
  24. self.per_page_num = per_page_num
  25.  
  26. # 实际总页码
  27. all_pager, tmp = divmod(all_count, per_page_num)
  28. if tmp:
  29. all_pager += 1
  30. self.all_pager = all_pager
  31.  
  32. self.pager_count = pager_count
  33. self.pager_count_half = int((pager_count - 1) / 2) #
  34.  
  35. # 保存搜索条件
  36.  
  37. import copy
  38. self.params=copy.deepcopy(request.GET) # {"a":"1","b":"2"}
  39.  
  40. @property
  41. def start(self):
  42. return (self.current_page_num - 1) * self.per_page_num
  43.  
  44. @property
  45. def end(self):
  46. return self.current_page_num * self.per_page_num
  47.  
  48. # 该方法用于在页面渲染分页的标签按钮
  49. def page_html(self):
  50. # 如果总页码 < 11个:
  51. if self.all_pager <= self.pager_count:
  52. pager_start = 1
  53. pager_end = self.all_pager + 1
  54. # 总页码 > 11
  55. else:
  56. # 当前页如果<=页面上最多显示11/2个页码
  57. if self.current_page_num <= self.pager_count_half:
  58. pager_start = 1
  59. pager_end = self.pager_count + 1
  60.  
  61. # 当前页大于5
  62. else:
  63. # 页码翻到最后
  64. if (self.current_page_num + self.pager_count_half) > self.all_pager:
  65.  
  66. pager_start = self.all_pager - self.pager_count + 1
  67. pager_end = self.all_pager + 1
  68.  
  69. else:
  70. pager_start = self.current_page_num - self.pager_count_half
  71. pager_end = self.current_page_num + self.pager_count_half + 1
  72.  
  73. page_html_list = [] # 存放分页标签,用于整体渲染
  74.  
  75. # 首页
  76. self.params["page"]=1 # 添加键值对,page=1
  77.  
  78. first_page = '<li><a href="?%s">首页</a></li>' % (self.params.urlencode(),)
  79. page_html_list.append(first_page)
  80.  
  81. # 上一页
  82. self.params["page"]=self.current_page_num - 1
  83.  
  84. if self.current_page_num <= 1:
  85. prev_page = '<li class="disabled"><a href="#">上一页</a></li>'
  86. else:
  87. prev_page = '<li><a href="?%s">上一页</a></li>' % (self.params.urlencode(),)
  88.  
  89. page_html_list.append(prev_page)
  90.  
  91. # 中间页
  92. for i in range(pager_start, pager_end):
  93.  
  94. self.params["page"]=i
  95.  
  96. if i == self.current_page_num:
  97. temp = '<li class="active"><a href="?%s">%s</a></li>' %(self.params.urlencode(),i)
  98. else:
  99. temp = '<li><a href="?%s">%s</a></li>' % (self.params.urlencode(),i,) # 保存搜索条件
  100. page_html_list.append(temp)
  101.  
  102. # 下一页
  103. self.params["page"]=self.current_page_num + 1
  104.  
  105. if self.current_page_num >= self.all_pager:
  106. next_page = '<li class="disabled"><a href="#">下一页</a></li>'
  107. else:
  108. next_page = '<li><a href="?%s">下一页</a></li>' % (self.params.urlencode(),)
  109. page_html_list.append(next_page)
  110.  
  111. # 尾页
  112. self.params["page"]=self.all_pager
  113.  
  114. last_page = '<li><a href="?%s">尾页</a></li>' % (self.params.urlencode(),)
  115. page_html_list.append(last_page)
  116.  
  117. return ''.join(page_html_list)

3、页面的渲染(html)

  借助bootstrap的样式

  1. <!DOCTYPE html>
  2. <html lang="zh-CN">
  3. <head>
  4. <meta charset="UTF-8">
  5. <title>Title</title>
  6. <meta name="viewport" content="width=device-width, initial-scale=1">
  7. <!-- 最新版本的 Bootstrap 核心 CSS 文件 -->
  8. <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">
  9. </head>
  10. <body>
  11.  
  12. <!-- 渲染数据库的内容 -->
  13. <ul>
  14. {% for book in book_list %}
  15. <li>{{ book.title }} ---- {{ book.price }}</li>
  16. {% endfor %}
  17. </ul>
  18.  
  19. <!-- 生成分页的标签 借助bootstrap的样式 nav ul-->
  20. <nav aria-label="Page navigation">
  21. <ul class="pagination">
  22.  
  23. {{ pagination.page_html|safe }}
  24.  
  25. </ul>
  26. </nav>
  27.  
  28. </body>
  29. </html>

自定义分页参考

一、分页的实现与使用

  1. class Pagination(object):
  2. """
  3. 自定义分页
  4. """
  5. def __init__(self,current_page,total_count,base_url,params,per_page_count=10,max_pager_count=11):
  6. try:
  7. current_page = int(current_page)
  8. except Exception as e:
  9. current_page = 1
  10. if current_page <=0:
  11. current_page = 1
  12. self.current_page = current_page
  13. # 数据总条数
  14. self.total_count = total_count
  15.  
  16. # 每页显示10条数据
  17. self.per_page_count = per_page_count
  18.  
  19. # 页面上应该显示的最大页码
  20. max_page_num, div = divmod(total_count, per_page_count)
  21. if div:
  22. max_page_num += 1
  23. self.max_page_num = max_page_num
  24.  
  25. # 页面上默认显示11个页码(当前页在中间)
  26. self.max_pager_count = max_pager_count
  27. self.half_max_pager_count = int((max_pager_count - 1) / 2)
  28.  
  29. # URL前缀
  30. self.base_url = base_url
  31.  
  32. # request.GET
  33. import copy
  34. params = copy.deepcopy(params)
  35. params._mutable = True
  36. # 包含当前列表页面所有的搜索条件
  37. # {source:[2,], status:[2], gender:[2],consultant:[1],page:[1]}
  38. # self.params[page] = 8
  39. # self.params.urlencode()
  40. # source=2&status=2&gender=2&consultant=1&page=8
  41. # href="/hosts/?source=2&status=2&gender=2&consultant=1&page=8"
  42. # href="%s?%s" %(self.base_url,self.params.urlencode())
  43. self.params = params
  44.  
  45. @property
  46. def start(self):
  47. return (self.current_page - 1) * self.per_page_count
  48.  
  49. @property
  50. def end(self):
  51. return self.current_page * self.per_page_count
  52.  
  53. def page_html(self):
  54. # 如果总页数 <= 11
  55. if self.max_page_num <= self.max_pager_count:
  56. pager_start = 1
  57. pager_end = self.max_page_num
  58. # 如果总页数 > 11
  59. else:
  60. # 如果当前页 <= 5
  61. if self.current_page <= self.half_max_pager_count:
  62. pager_start = 1
  63. pager_end = self.max_pager_count
  64. else:
  65. # 当前页 + 5 > 总页码
  66. if (self.current_page + self.half_max_pager_count) > self.max_page_num:
  67. pager_end = self.max_page_num
  68. pager_start = self.max_page_num - self.max_pager_count + 1 #倒这数11个
  69. else:
  70. pager_start = self.current_page - self.half_max_pager_count
  71. pager_end = self.current_page + self.half_max_pager_count
  72.  
  73. page_html_list = []
  74. # {source:[2,], status:[2], gender:[2],consultant:[1],page:[1]}
  75. # 首页
  76. self.params['page'] = 1
  77. first_page = '<li><a href="%s?%s">首页</a></li>' % (self.base_url,self.params.urlencode(),)
  78. page_html_list.append(first_page)
  79.  
  80. # 上一页
  81. self.params["page"] = self.current_page - 1
  82. if self.params["page"] < 1:
  83. pervious_page = '<li class="disabled"><a href="%s?%s" aria-label="Previous">上一页</span></a></li>' % (
  84. self.base_url, self.params.urlencode())
  85. else:
  86. pervious_page = '<li><a href = "%s?%s" aria-label = "Previous" >上一页</span></a></li>' % (
  87. self.base_url, self.params.urlencode())
  88. page_html_list.append(pervious_page)
  89.  
  90. # 中间页码
  91. for i in range(pager_start, pager_end + 1):
  92. self.params['page'] = i
  93. if i == self.current_page:
  94. temp = '<li class="active"><a href="%s?%s">%s</a></li>' % (self.base_url,self.params.urlencode(), i,)
  95. else:
  96. temp = '<li><a href="%s?%s">%s</a></li>' % (self.base_url,self.params.urlencode(), i,)
  97. page_html_list.append(temp)
  98.  
  99. # 下一页
  100. self.params["page"] = self.current_page + 1
  101. if self.params["page"] > self.max_page_num:
  102. self.params["page"] = self.current_page
  103. next_page = '<li class="disabled"><a href = "%s?%s" aria-label = "Next">下一页</span></a></li >' % (
  104. self.base_url, self.params.urlencode())
  105. else:
  106. next_page = '<li><a href = "%s?%s" aria-label = "Next">下一页</span></a></li>' % (
  107. self.base_url, self.params.urlencode())
  108. page_html_list.append(next_page)
  109.  
  110. # 尾页
  111. self.params['page'] = self.max_page_num
  112. last_page = '<li><a href="%s?%s">尾页</a></li>' % (self.base_url, self.params.urlencode(),)
  113. page_html_list.append(last_page)
  114.  
  115. return ''.join(page_html_list)

二、使用

  1. """
  2. 自定义分页组件的使用方法:
  3. pager_obj = Pagination(request.GET.get('page',1),len(HOST_LIST),request.path_info,request.GET)
  4. host_list = HOST_LIST[pager_obj.start:pager_obj.end]
  5. html = pager_obj.page_html()
  6. return render(request,'hosts.html',{'host_list':host_list,"page_html":html})
  7. """

三、批量查数据

  1. #批量创建数据
  2. # booklist = []
  3. # for i in range(100):
  4. # booklist.append(models.Book(name="book" + str(i), price=20 + i * i))
  5. #
  6. # models.Book.objects.bulk_create(booklist)

Django的用户认证组件,自定义分页的更多相关文章

  1. python 全栈开发,Day79(Django的用户认证组件,分页器)

    一.Django的用户认证组件 用户认证 auth模块 在进行用户登陆验证的时候,如果是自己写代码,就必须要先查询数据库,看用户输入的用户名是否存在于数据库中: 如果用户存在于数据库中,然后再验证用户 ...

  2. django - 总结 - 用户认证组件

    用户认证组件 from django.contrib import auth 从auth_user表中获取对象,没有返回None,其中密码为密文,使用了加密算法 user = auth.authent ...

  3. django的用户认证组件

    DataSource:https://www.cnblogs.com/yuanchenqi/articles/9064397.html 代码总结: 用户认证组件: 功能:用session记录登录验证状 ...

  4. 06 django的用户认证组件

    1.用户认证组件 用户认证组件: 功能:用session记录登录验证状态 前提:用户表:django自带的auth_user 创建超级用户: python3 manage.py createsuper ...

  5. Django之用户认证组件

    auth模块 之前我们在进行用户登录验证的时候,都是自己写代码,接收用户提交的数据,然后去数据库取数据进行匹配验证,其实Django已经给我们提供了内置的用户认证功能.不信的话你可以打开models. ...

  6. 14 Django的用户认证组件

    用户认证 auth模块 from django.contrib import auth django.contrib.auth中提供了许多方法,这里主要介绍其中的三个: 1.1 .authentica ...

  7. Django学习笔记(13)——Django的用户认证(Auth)组件,视图层和QuerySet API

    用户认证组件的学习 用户认证是通过取表单数据根数据库对应表存储的值做比对,比对成功就返回一个页面,不成功就重定向到登录页面.我们自己写的话当然也是可以的,只不过多写了几个视图,冗余代码多,当然我们也可 ...

  8. Django:(7)auth用户认证组件 & 中间件

    用户认证组件 用户认证组件: 功能:用session记录登陆验证状态 前提:用户表:django自带的auth_user 创建超级用户的命令: python manage.py createsuper ...

  9. web框架开发-Django用户认证组件

    可以用认证组件做什么 针对session的缺陷, 跟新数据时,不跟新key键, 用户认证组件是删除后再重建 用户认证组件很多功能可以直接使用 利用用户认证表(auth_user,通过Django自己创 ...

随机推荐

  1. JAVA JDBC 读取配置文件链接数据库(oracle)

    ----db.properties-------- dbDriver = oracle.jdbc.driver.OracleDriverurl = jdbc:oracle:thin:@192.168. ...

  2. 分立元件封装尺寸及PCB板材工艺与设计实例

    分立元件封装尺寸 inch mm (L)mm (w)mm (t)mm (a)mm (b)mm 0201 0603 0.6±0.05 0.30±0.05 0.23±0.05 0.10±0.05 0.60 ...

  3. mesos的zookeeper变更

    采用rpm方式安装你了mesos,碰到zookeeper(采用了cloudera的zookeeper)的IP地址变化了,肿么办? 在master机器中: /etc/mesos/zk进行编辑修改zk路径 ...

  4. Java中的String数据类型

    本文主要是说明一些String数据类型的基本知识,有些杂乱,不过都是比较重要的东西,主要是参考了网上人的资料.原文网址:http://dev.yesky.com/91/2309091.shtml 主要 ...

  5. redis多机集群部署文档

    redis多机集群部署文档(centos6.2) (要让集群正常工作至少需要3个主节点,在这里我们要创建6个redis节点,其中三个为主节点,三个为从节点,对应的redis节点的ip和端口对应关系如下 ...

  6. cmake opencv,dlib 编译静态库 1

    无论windows,linux 所有的库 ,dlib,opencv 通过cmake-gui 设置好静态库, 动态库,和其他各个选项 Tips: cmake 优先级用cmake-gui,因为命令太多,容 ...

  7. hibernate HQL查询

    hql(都要在事务中完成)session.beginTransaction();session.getTransaction().commit(); session.beginTransaction( ...

  8. 1.JasperReports学习笔记1-了解JasperReports

    转自:http://www.blogjava.net/vjame/archive/2013/10/12/404908.html JasperReports是一个开源的java报表制作引擎,官网地址:h ...

  9. day02 秘钥生成,免密访问命令

    hadoop免密登陆: 生成秘钥: ssh-keygen -t dsa -P '' -f ~/.ssh/id_dsa cat ~/.ssh/id_dsa.pub >> ~/.ssh/aut ...

  10. mongodb 操作数据库

    1.使用和创建数据库: use mydb //没有就创建 2.显示数据库 show dbs 3.显示数据库状态 db.stats() 4.检查当前所用的数据库 db 5.删除数据库(先用然后删除) u ...