django的forms认证组件

每个网站的注册界面都需要有相应的“认证”功能,比如说认证注册页面的用户名是否已被注册,二次输入的密码是否一致以及认证用户输入的用户名、邮箱、手机号等字段是否符合程序员所指定的“规则”等等。当然,不同的web框架有不同的实现方式,本文介绍django的forms组件的实现方法。

默认的校验规则

默认的校验规则都是校验单个字段而且格式是固定的:

建表

models.py文件为数据库中创建下面字段:
from django.db import models
class UserInfo(models.Model):
    name = models.CharField(max_length=32)
    pwd = models.CharField(max_length=32)
    email = models.EmailField()
    tel = models.CharField(max_length=32)

模板文件

模板文件我们引用bootstrap,并且利用模板语言进行渲染得到相应的页面:
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Forms组件</title>
    <link rel="stylesheet" href="/static/bootstrap-3.3.7/css/bootstrap.css">
    <link rel="stylesheet" href="/static/whw_css.css">
</head>
<body>
<h3>注册页面</h3>
<div class="container">
    <div class="row">
        <div class="col-md-6 col-lg-offset-3">
           <form action="" method="post" novalidate >
            {% csrf_token %}
            <p>用户名 {{ form.name }}<span class="pull-right">{{ form.name.errors.0 }}</span></p>
            <p>密码 {{ form.pwd }}<span class="pull-right">{{ form.pwd.errors.0 }}</span></p>
            <p>确认密码 {{ form.r_pwd }}<span class="pull-right">{{ form.r_pwd.errors.0 }}</span><span class="pull-right">{{ full_errors.0 }}</span></p>
            <p>邮箱 {{ form.email }}<span class="pull-right">{{ form.email.errors.0 }}</span></p>
            <p>手机号 {{ form.tel }}<span class="pull-right">{{ form.tel.errors.0 }}</span></p>
            <input type="submit" value="确认">
        </form>
        </div>
    </div>
</div>
</body>
</html> 

校验模块

为了使程序易于维护且解耦性高一点,我们把认证功能写在一个模块my_forms.py里,在里面定义一个专门用于认证的类UserForm:
#先引入相应的模块
from django import forms
from django.forms import widgets
from app01.models import UserInfo
from django.core.exceptions import NON_FIELD_ERRORS,ValidationError

#做校验的类,后面跟着校验规则
class UserForm(forms.Form):
    ##校验字段
    name = forms.CharField(min_length=4,label='用户名',widget=widgets.TextInput(attrs={'class':'form-control'}),error_messages={'required':'输入不能为空'})

    email = forms.EmailField(label='邮箱',widget=widgets.TextInput(attrs={'class':'form-control'}),error_messages={'required':'输入不能为空','invalid':'请输入邮箱格式'})

    pwd = forms.CharField(min_length=4,label='密码',widget=widgets.PasswordInput(attrs={'class':'form-control'}),error_messages={'required':'输入不能为空'})

    r_pwd = forms.CharField(min_length=4,label='确认密码',widget=widgets.PasswordInput(attrs={'class':'form-control'}),error_messages={'required':'输入不能为空'})

    tel = forms.CharField(label='手机号',widget=widgets.TextInput(attrs={'class':'form-control'}),error_messages={'required':'输入不能为空'})
参数说明:
(1)min_length=4表示输入的字符的长度不能小于4
(2)widget=widgets.TextInput(attrs={'class':'form-control'}) 这句话表示输入框input的type='text',且class='form-control'
(3)error_messages是在用户输入不符合校验规则时做出的说明,是一个字典的形式。注意字典里的key值是固定的!

视图函数

views.py文件的内容如下:
from django.shortcuts import render,HttpResponse
##引入校验的类
from app01.my_forms import UserForm

#视图函数
def my_forms(request):
    if request.method == 'GET':
        form = UserForm()
        return render(request, 'index.html',{'form':form})

    if request.method == 'POST':
        #form表单的name属性值应当与forms组件的字段名称一致
        form = UserForm(request.POST)
        print(form.is_valid())
        if form.is_valid():
            print(form.cleaned_data)
            #只有全部验证成功了才执行
            return HttpResponse('OK!')
        else:
            print(form.errors)
            #带着参数返回当前页面
            return render(request,'index.html',locals())
如上面的代码所示,我们用UserForm类以及前端返回的参数实例化出一个类form:
form = UserForm(request.POST)
这里有一个is_valid()方法,返回布尔类型的值。只有当全部字段校验成功后它的值才为True。
最终校验成功的字段存到了form.cleaned_data里;校验失败的字段以及对应的错误信息放在了form.errors里。将这两个“字典”类型的数据传给模板我们就可以看到对应的信息了:

自定义校验规则与多个字段的校验

一、自定义校验规则我们用到局部钩子。如果我们想校验属性name,那么可以在UserForm类中我们定义一个方法,名字固定为clean_name
##校验数据库中是否有相同的用户名
def clean_name(self):
    val = self.cleaned_data.get('name')
    #数据库中检测
    ret = UserInfo.objects.filter(name=val)
    if not ret:
        return val
    #如果ret有值表示数据库中有这条记录了
    else:
        raise ValidationError('该用户已注册')

注意这个方法名固定,必须是clean_%s的形式,这里的%s就是校验的属性的名字。比如说我们想校验手机号必须为11位可以这样写:
def clean_tel(self):
    val = self.cleaned_data.get('tel')
    if len(val) == 11:
        return val
    else:
        raise ValidationError('手机号必须为11位')

二、进行多个字段的校验我们用到全局钩子
如果我们想校验两次收入的密码是否一致可以在UserForm中定义一个方法clean:
def clean(self):
    pwd = self.cleaned_data.get('pwd')
    r_pwd = self.cleaned_data.get('r_pwd')
    if pwd and r_pwd:
        if pwd == r_pwd:
            return self.cleaned_data
        else:
            raise ValidationError('两次密码不一致')
    else:
        return self.cleaned_data

django的forms认证组件的更多相关文章

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

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

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

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

  3. Django的rest_framework认证组件之全局设置源码解析

    前言: 在我的上一篇博客我介绍了一下单独为某条url设置认证,但是如果我们想对所有的url设置认证,该怎么做呢?我们这篇博客就是给大家介绍一下在Rest_framework中如何实现全局的设置认证组件 ...

  4. Django的rest_framework认证组件之局部设置源码解析

    前言: Django的rest_framework组件的功能很强大,今天来我来给大家剖析一下认证组件 下面进入正文分析,我们从视图开始,一步一步来剖析认证组件 1.进入urls文件 url(r'^lo ...

  5. django的用户认证组件

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

  6. 06 django的用户认证组件

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

  7. Django的用户认证组件,自定义分页

    一.用户认证组件 1.auth模块 from django.conrtrib import auth django.contrib.auth中提供了许多方法,这里主要介绍其中的三个: 1)authen ...

  8. Django之REST_FRAMEWORK 认证组件

    Django之DRF之认证组件 # from rest_framework.views import APIView # APIView 中的 dispatch 中 执行的 self.initial( ...

  9. Django REST framework —— 认证组件源码分析

    我在前面的博客里已经讲过了,我们一般编写API的时候用的方式 class CoursesView(ViewSetMixin,APIView): pass 这种方式的有点是,灵活性比较大,可以根据自己的 ...

随机推荐

  1. restore not found的错误

    tensorflow保存模型后,restore的时候报参数not found是什么原因呢 一般预测的流程是:建图然后restore参数,很有可能你的变量作用域和train的时候不一样,那么在现在的变量 ...

  2. 微信小程序城市定位(百度地图API)

    概述 微信小程序提供一些API(地址)用于获取当前用户的地理位置等信息,但无论是wx.getLocation,还是wx.chooseLocation均没有单独的字段表示国家与城市信息,仅有经纬度信息. ...

  3. 如何解析超长的protobuf zhuan

    在调用protobuf的ParseFromString(str)方法时,默认情况下,如果str的长度>64MB,会返回失败. 这里给出了解释,主要是出于安全因素的考虑. 可以通过SetTotal ...

  4. Java 代理

    代理做一个简单的抽象: 代理模式包含如下角色: Subject:抽象主题角色.可以是接口,也可以是抽象类. RealSubject:真实主题角色.业务逻辑的具体执行者. ProxySubject:代理 ...

  5. C++设计模式之-代理模式

    根据程洁的大话模式: // Proxy.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include<iostream> # ...

  6. Python选修课第一届Turtle绘图大赛田康林赵冰珂组

    点击此处查看视频 from turtle import* setup(600,600,200,200) #脸 penup() goto(-190,0) seth(-90) pendown() penc ...

  7. Problem F: 平面上的点——Point类 (VI)

    Description 在数学上,平面直角坐标系上的点用X轴和Y轴上的两个坐标值唯一确定.现在我们封装一个“Point类”来实现平面上的点的操作. 根据“append.cc”,完成Point类的构造方 ...

  8. 测试同学必备抓包工具--charles之安装

    1,下载charles,官网:https://www.charlesproxy.com/ 2,下载完成,先试着用一下,网址访问百度看看... 注意,windows proxy如果勾选,则代表可以抓取网 ...

  9. 获取当前时间并格式化,CTime类

    CTime类,此类应该不是C++标准类库,属于windows封装的关于时间的类库,使用环境应该为 Win32程序,MFC程序,VC++程序 CTime tm = CTime::GetCurrentTi ...

  10. FCC JS基础算法题(13):Caesars Cipher(凯撒密码)

    题目描述: 下面我们来介绍风靡全球的凯撒密码Caesar cipher,又叫移位密码.移位密码也就是密码中的字母会按照指定的数量来做移位.一个常见的案例就是ROT13密码,字母会移位13个位置.由'A ...