我在之前做了一个关于AJAX和form组件的笔记,可以参考:Django学习笔记(8)——前后台数据交互实战(AJAX)Django学习笔记(6)——Form表单

 

  我觉得自己在写Django笔记(8)的时候,我只是对AJAX有个粗略的了解,就明白其两大优点,局部刷新和异步交互。Form组件知识虽然大多数都明白了,但是对局部钩子和全局钩子还不是很清楚。留了个坑,所以在以后的学习中,肯定会再遇到。

  现在,我感觉自己将关于AJAX和Django的数据交互这部分了解清楚了,而且将Form组件的钩子也理解透彻了。特意做个笔记记录自己的学习过程。也是在Django笔记(8)的时候说过的,我将这段内容理清楚了,会写一个易于理解的文章巩固,而这篇文件就是。

  下面我分别将自己所做的AJAX知识笔记,Form组件中局部钩子和全局钩子的笔记展示一下。其中局部钩子和全局钩子都做了源码分析,是为了更好的理解其原理吧。

 

AJAX知识

1,AJAX请求头中常见contentType

  数据发送出去,还需要服务端解析成功才有意义。Python内置了自动解析常见数据格式的功能。服务端通常是根据请求头(headers)中的Content-Type 字段获取请求头中的消息主体是用何种方式编码,再对主体进行解析。所以说到POST提交数据方案,包含了Content-Type 和消息主体编码方式两部分。

  下面我们一起来看看ajax中POST请求的Content-Type。以application开头的媒体格式类型:

application/xhtml+xml :XHTML格式

application/xml     : XML数据格式

application/atom+xml  :Atom XML聚合格式   

application/json    : JSON数据格式

application/pdf       :pdf格式 

application/msword  : Word文档格式

application/octet-stream : 二进制流数据(如常见的文件下载)

application/x-www-form-urlencoded : 
<form encType=””>中默认的encType,form表单数据被编码为key/value格式
发送到服务器(表单默认的提交数据的格式)

  ContentType指的是请求体的编码类型,常见的有三种:

1.1,application/x-www-form-urlencoded

  这应该是最常见的POST提交数据的方式了。浏览器的原生表单<form>默认的提交方式,如果不设置enctype属性,那么最终就会以application/x-www-form-urlencoded 方式提交数据。请求类似于下面这样(无关的请求头在本文中都省略掉了):

POST http://www.example.com HTTP/1.1
Content-Type: application/x-www-form-urlencoded;charset=utf-8

user=james&age=24

  

  下面将参数的默认使用方法总结如下:

data:        当前ajax请求要携带的数据,是一个object对象,ajax方法就会默认地把
它编码成某种格式(urlencoded:?a=1&b=2)发送给服务端;此外,ajax默认以get方
式发送请求。
 
contentType:"application/x-www-form-urlencoded"。发送信息至服务器时内容
编码类型。用来指明当前请求的数据编码格式;urlencoded:?a=1&b=2;

  

 

1.2,multipart/form-data

  这又是一个常见的POST数据提交的方式。我们使用表单上传文件时,必须让表单的enctype 等于 multipart/form-data。

  直接看一个例子:

POST http://www.example.com HTTP/1.1
Content-Type:multipart/form-data; boundary=----WebKitFormBoundaryrGKCBY7qhFd3TrwA

------WebKitFormBoundaryrGKCBY7qhFd3TrwA
Content-Disposition: form-data; name="user"

james
------WebKitFormBoundaryrGKCBY7qhFd3TrwA
Content-Disposition: form-data; name="file"; filename="chrome.png"
Content-Type: image/png

PNG ... content of chrome.png ...
------WebKitFormBoundaryrGKCBY7qhFd3TrwA--

  这个例子稍微复杂点,首先生成了一个boundary 用于分割不同的字段,为了避免与正文内容重复,boundary很长很复杂。然后Content-Type里指明了数据是以 multipart/form-data 来编码。

  本次请求的 boundary 是什么内容?

  消息主体里是按照字段个数又分为多个结构类似的部分,每部分都是以 –boundary 开始。紧接着是内容描述信息,然后是回车,最后是字段具体内容(文本或者二进制)。如果传输的是文件,还要包含文本名和文件类型信息。消息主体最后以 –boundary– 标识结束。关于 multipart/form-data 的详细定义,请前往 rfc 1967 查看。

  这种方式一般用来上传文件,各大服务端语言对它也有着良好的支持。

  上面提到的这两种POST数据的方式,都是浏览器原生支持的,而且现阶段标准中原生表单也只支持这两种方式。其实enctype 还支持 text/plain,不过用的非常少。

  随着越来越多的Web站点,尤其是WebAPP,全部使用AJAX进行数据交互之后,我们完全可以定义新的数据提交方式,给开发带来更多便利。

 

1.3,application/json

  application/json 这个Content-Type 作为响应头大家肯定不陌生。实际上,现在越来越多的人吧他作为请求头,用来告诉服务端消息主体是序列化的JSON字符串。由于JSON规范的流行,处理低版本IE之外的各大浏览器都支持原生的JSON.stringify,服务端语言也有处理JSON的函数,使用JSON不会遇到什么麻烦。

 

  上述这种默认参数形式,data中的csrf跨站请求伪造键值对会被中间件自动识别,ContentType参数还有如下一种形式,介绍如下:

contentType:"application/json",即向服务器发送一个json字符串。
 
注意:contentType:"application/json"一旦设定,data必须是json字符串,不能是json对象

  这种类型,使用request.POST 无法显示,这种类型要使用 request.body才能显示数据。

1.4,text/xml(不常用)

  相比于JSON,XML不能更好的适用于数据交换,它包含了太多的包装,而且它跟大多数编程语言的数据模型不匹配,让大多数程序员感到诧异,XML是面向数据的,JSON是面向对象和结构的,后者会给程序员一种更加亲切的感觉。

  我们现在一般这样来使用:

  • 1,XML存储数据,存储配置文件等需要结构化存储的地方使用;
  • 2,数据传输,数据交互使用JSON

  下面就是 ajax Content-Type 为 text/xml 的请求:

 

2,文件上传

  注意:我们使用表单上传文件时,必须让表单的enctype 等于 multipart/form-data。

2.1  基于form表单的文件上传

  html代码:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>upload</title>
</head>
<body>
 
{# 文件上传的form#}
{#<form method="post" enctype="multipart/form-data">#}
<form method="post" action="/user8_book/upload.html/" enctype="multipart/form-data">
    {% csrf_token %}
    <p><input id="name" type="text"  name="input-name" placeholder="请输入文件名称" ></p>
    <p><input id="file" type="file" name="file"></p>
    <p><input id="submit" type="submit" value="submit"></p>
</form>
 
</body>
</html>

  view视图函数:

from django.shortcuts import render, HttpResponse
import os

# Create your views here.

def index(request):
    if request.method == \'POST\':
        obj = request.FILES.get(\'file\')
        print("获取文件的名称: ", obj.name)

        f = open(os.path.join(\'statics/upload\', obj.name), \'wb\')
        for line in obj.chunks():
            f.write(line)
        f.close()

        # POST请求数据
        print("POST 请求数据: ", request.POST)
        # GET 请求数据
        print("上传的文件数据: ", request.FILES)

        return HttpResponse("upload OK")

    return render(request, \'file_put/index.html\')

  我们上传数据后,在后台终端可以看到下面信息:

获取文件的名称:  distance.jpg

POST 请求数据:  <QueryDict: {\'csrfmiddlewaretoken\': 
[\'uHoRIy1DrQyS2nD87UndwfoQXH9KEd5zEcIpzjZPjo2zk84TfBatR9QCbaAmckFh\'], 
\'user\': [\'file1\']}>

上传的文件数据:  <MultiValueDict: {\'avatar\': [<InMemoryUploadedFile: distance.jpg 
(image/jpeg)>]}>

  

2.2 基于AJAX文件上传

  html代码:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    
    <script src="http://ajax.aspnetcdn.com/ajax/jquery/jquery-2.1.4.min.js"></script>
    
</head>
<body>

    <h2>基于AJAX文件上传</h2>
    <form action="" method="post" enctype="multipart/form-data">
         {% csrf_token %}
        <p>username:<input type="text" name="user"  placeholder="请输入文件名称"></p>
        <p>photo:<input type="file" name="avatar" ></p>
        <p><input type="submit" value="ajax submit" class="btn"></p>
    </form>
</body>

<script>
    //  ajax上传
    $(\'.btn\').click(function () {
        var formdata = new FormData();
        formdata.append(\'user\', $("#user").val());
        formdata.append(\'avatar\', $("#avatar")[0].files[0]);

        $.ajax({
            url: \'\',
            type: \'post\',
            contentType: false,
            processData: false,
            data: formdata,
            success: function (data){
                console.log(data)
            }
        })
    })
</script>
</html>

  注意1:使用ajax上传文件的时候,前端代码必须出现下面:

            contentType: false,
            processData: false,

  注意2:AJAX(Formdata)是什么呢?

  XMLHttpRequest Level 2 添加了一个新的接口 FormData,利用FormData对象,我们可以通过JavaScript用一些键值对来模拟一系列表单控件,我们还可以使用XMLHttpRequest 的 send()  方法来异步的提交这个“表单”。比起普通的 ajax,使用FormData的最大优点就是我们可以异步上传一个二进制文件。

  所有主流浏览器的较新版本都已经支持这个对象了。比如Chtome 7+, Firefox 4+, IE 10+,Opera 12+,Safari 5+ 等。

   view函数:

from django.shortcuts import render, HttpResponse
import os

# Create your views here.

def index(request):
    if request.method == \'POST\':
        obj = request.FILES.get(\'avatar\')
        print("获取文件的名称: ", obj.name)

        f = open(os.path.join(\'statics/upload\', obj.name), \'wb\')
        for line in obj.chunks():
            f.write(line)
        f.close()

        # POST请求数据
        print("POST 请求数据: ", request.POST)
        # GET 请求数据
        print("上传的文件数据: ", request.FILES)

        return HttpResponse("upload OK")

    return render(request, \'file_put/index.html\')

def ajax_put(request):
    if request.method == \'POST\':
        obj = request.FILES.get(\'avatar\')
        print("获取文件的名称: ", obj)
        # POST请求数据
        print("POST 请求数据: ", request.POST)
        # 文件的请求数据
        print("上传的文件数据: ", request.FILES)
        f = open(os.path.join(\'statics/upload\', obj.name), \'wb\')
        for line in obj.chunks():
            f.write(line)
        f.close()

        return HttpResponse("upload OK")

    # if request.method == \'POST\':
    #     # 请求报文中的请求体
    #     print(request.body)
    #     print(request.POST)
    #     print(request.FILES)
    #     file_obj = request.FILES.get(\'avatar\')
    #     with open(file_obj.name, \'wb\') as f:
    #         for line in file_obj:
    #             f.write(line)
    #     return HttpResponse("AJAX OK")

    return render(request, \'file_put/ajax_index.html\')

  我们上传数据后,在后台终端可以看到下面信息:

获取文件的名称:  50788990.jpg

POST 请求数据:  <QueryDict: {\'csrfmiddlewaretoken\': 
[\'x1KicPbnZ6k7lg7AeerN4WBfUC14JLeoHw4Q3A9zREOOD1ylmVe3pQ3185sGhSO6\'], 
\'user\': [\'file132\']}>

上传的文件数据:  
<MultiValueDict: {\'avatar\': [<InMemoryUploadedFile: 50788990.jpg (image/jpeg)>]}>

  

Form组件学习

1,什么是forms组件

  forms组件就是一个类,可以检查前端传来的数据,是否合法。

  例如前端传来的邮箱数据,判断邮箱格式是否正确,用户名不能以什么开头等等。

2,form组件的使用语法

  简单举个例子说一下:

from django.shortcuts import render, HttpResponse
from django import forms

# 1.先写一个类,继承Form
class MyForm(forms.Form):
    # 定义一个属性,可以用来校验字符串类型
    # 限制最大长度是8,最小长度是3
    name=forms.CharField(max_length=8,min_length=3)
    pwd=forms.CharField(max_length=8,min_length=3,required=True)
    # 校验是否是邮箱格式
    email=forms.EmailField()

    

# 2.在视图函数中使用MyForm来校验数据
# 实例化产生对象,传入要校验的数据(可以传字典字典,也可以不传)
myform=MyForm(request.POST)

# 3.校验,is_valid如果是true表示校验成功(满足myform里的条件),反之,校验失败
if myform.is_valid():
    # myform.clean_data 表示校验通过的数据
    print(myform.cleaned_data)
    return HttpResponse(\'校验成功\')
else:
    print(myform.cleaned_data)
    #校验失败的信息,myform.errors  可以当成一个字典,它是所有错误信息{name:[列表,]}
    # 每个字段.errors 是一个列表,表示每个字段的错误信息
    print(myform.errors)
    return HttpResponse(\'校验失败\')

  方法总结:

  • myform.clean_data 验证通过的数据
  • myform.errors 错误数据的对象
  • myform.errors.as_data 错误数据的信息

3,渲染模板

  form组件可以在视图函数中使用,也可以在前端模板中使用。我们展示一下视图层和模板层。

视图层:

# 视图层:
def index(request):
    myform = Myform()
    return render(request,\'index.html\',local())

  

模板层:

# 模板层
# 1.渲染方式一:
    <form action=\'\' method=\'post\'>
        用户名:{{myform:name}} <br>
        <input type=\'submit\' value = \'提交\'></input>
    </form>
    # 这里的{{myform:name}} 和你写input框是一样的效果,就是属性比input框多一点

# 2.渲染方式二(推荐使用):
    <form action=\'\' method=\'post\'>
        {% for foo in myform%}
            {{ foo.lable }} : {{ foo }}  <br>
        <input type=\'submit\' value = \'提交\'></input>
    </form>
    # 页面显示都是一样的,foo.lable不是用户名,是name,但是可以在创建Myform类时,在CharFiel中添加lable=\'用户名\',这样就行了。

# 3.渲染方式三:
    <form action=\'\' method=\'post\'>
         {{ myform.as_p }}
        <input type=\'submit\' value = \'提交\'></input>
    </form>

  下面举3个例子,详细说一下这三种渲染方式

3.1  form组件的渲染方式1

  html代码如下:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>

<form action="" method="post">
    {% csrf_token %}
    <p>username: <input type="text" name="name"></p>
    <p>password:<input type="text" name="pwd"></p>
    <p>re_password:<input type="text" name="r_pwd"></p>
    <p>email:<input type="text" name="email"></p>
    <p>telephone: <input type="text" name="tel"></p>
    <p><input type="submit"></p>
</form>

<hr>
<h3>渲染方式1 form组件的渲染</h3>
<form action="" method="post">
    {% csrf_token %}
    <p>username:
        {{ form.name }}
    </p>
    <p>password:
        {{ form.pwd }}
    </p>
    <p>re_password:
        {{ form.r_pwd }}
    </p>
    <p>email:
        {{ form.email }}
    </p>
    <p>telephone:
        {{ form.tel }}
    </p>
    <p><input type="submit"></p>
</form>

</body>
</html>

  views.py

from django.shortcuts import render,HttpResponse

# Create your views here.

from django import forms

class UserForm(forms.Form):
    name = forms.CharField(min_length=3)
    pwd = forms.CharField(min_length=3)
    r_pwd = forms.CharField(min_length=3)
    email = forms.EmailField()
    # 手机号码的规则固定为11位,***
    tel = forms.CharField(min_length=3)

def reg(request):
    if request.method == \'POST\':
        # print(request.POST)
        # form = UserForm({\'name\': \'yuan\', \'email\': \'123\'})
        # print(form.is_valid())

        form = UserForm(request.POST)
        print(form.is_valid())
        if form.is_valid():
            print(form.cleaned_data)
        else:
            print(form.errors)


        return HttpResponse("OK")

    form = UserForm()

    return render(request, \'form/reg.html\', locals())

  

3.2  form组件渲染方式2

  如果变量太多了,我们不可能这样写很多个,所以我们使用for循环。

<hr>
<h3>渲染方式2 form组件的渲染</h3>
<form action="" method="post">
    {% csrf_token %}
    {% for foo in form  %}
        <div>
            <label for="">{{ foo.label }}</label>
            {{ foo }}
        </div>
    {% endfor %}

    <p><input type="submit"></p>
</form>

  我们可以在form函数里面设置label标签:

class UserForm(forms.Form):
    name = forms.CharField(min_length=3, label=\'用户名\')
    pwd = forms.CharField(min_length=3, label=\'密码\')
    r_pwd = forms.CharField(min_length=3, label=\'确认密码\')
    email = forms.EmailField( label=\'邮箱\')
    # 手机号码的规则固定为11位,***
    tel = forms.CharField(min_length=3, label=\'电话\')

  

3.3  form组件渲染方式3

<hr>
<h3>渲染方式3 form组件的渲染</h3>
<form action="" method="post">
    {% csrf_token %}
    {{ form.as_p }}

    <p><input type="submit"></p>
</form>

  但是这种方式不推荐使用。因为其固定死了格式。简单测试的时候可以使用。

   三种效果都一样,结果也是,这里就不再展示了。我们推荐使用第二种渲染方式。

4,form组件渲染错误信息

   当返回请求的数据失败的时候,我们如何渲染错误信息

<form action=\'\' method=\'post\'>
    {% for foo in myform%}
        {{ foo.lable }} : {{ foo }} <span>{{foo.errors.0}}</span><br>
    <input type=\'submit\' value = \'提交\'></input>
</form>

 举个例子:

  视图函数:

def register(request):

    if request.method=="POST":
        form=UserForm(request.POST)
        if form.is_valid():
            print(form.cleaned_data)       # 所有干净的字段以及对应的值
        else:
            print(form.cleaned_data)       #
            print(form.errors)             # ErrorDict : {"校验错误的字段":["错误信息",]}
            print(form.errors.get("name")) # ErrorList ["错误信息",]
        return render(request,"register.html",locals())
    form=UserForm()
    return render(request,"register.html",locals())

  模板:

<form action="" method="post" novalidate>
    {% csrf_token %}
    
    {% for field in form %}
        <div>
            <label for="">{{ field.label }}</label>
            {{ field }} <span class="pull-right" style="color: red">{{ field.errors.0 }}</span>
        </div>
    {% endfor %}
    <input type="submit" class="btn btn-default">

</form>

  

 

5,Form组件校验的局部钩子和全局钩子

5.1,什么是局部钩子

  定义一个函数,名字叫:clean_字段名字,内部,取出该字段,进行校验。如果通过,将该字段返回,如果失败,抛出异常(ValidationError)。

  其中ValidationError的异常类型需要引入:

from django.core.exceptions import ValidationError

  

5.2,局部钩子的源码分析

   首先,我们分析一段源码。关于钩子函数:

  从is_valid()点进去,然后点击errors:(也就是 form/forms.py的源码)

我们可以看到下面:

  点进去full_clean(),然后查看_clean_filed_() 方法

  注意(看源码诀窍):看源码的时候,看不懂就过!!

  这里我们查看_clean_fields()函数,这是局部钩子的应用

   首先,从fileds字段中导入其内容,当没有问题的时候,尝试判断其是否合法,如果合法的话,我们将vlaue字段赋值给 cleaned_data[name],然后利用一个反射函数,尝试调用局部钩子,如果有局部钩子函数,我们调用并执行,如果校验成功,我们将name 的值返回到clean_data,并写入clean_data字典中,也就是更新字典,如果出错的话,就抛出异常,将异常信息以键值对({\’name\’: value} 写入 errors 字典中)。

  如果出错的话,局部钩子抛出的异常会添加到该字段中的错误信息,我们在前台获取错误信息的方法如下:

    for循环生成 input 框

    {{ foo.errrs.0 }}

  

5.3,局部钩子示例:

 # 函数名称必须以 claen_字段名  的格式
    def clean_name(self):
        # 目的:如果用户名已经注册,则报异常
        val = self.cleaned_data.get(\'name\')

        ret = UserInfo.objects.filter(name=val)
        if not ret:
            return val
        else:
            raise ValidationError(\'用户名已经被注册\')

    def clean_tel(self):
        # 目的:校验手机号码长度为11
        val = self.cleaned_data.get(\'tel\')

        if len(val) == 11:
            return val
        else:
            raise ValidationError("手机号码格式错误")

5.4,什么是全局钩子

  在写注册用户的时候,有输入密码,确认密码,可以进行布局钩子处理,处理完毕是不是在进行判断,判断其是否相等,相等的话就存到数据库中,不相等就抛出异常。

  全局钩子主要应用场景就是每次校验多个字段,而局部钩子每次取的是单个字段,单个变量。

5.5,全局钩子的源码分析

  下面继续分析一段源码。

  从cleaned_data() 点击进去,我们会发现 clean_form执行的方法是 clean()方法。

  下面我们看clean()方法:

  我们会发现 默认的clean()方法什么都没有写,直接返回的一个 cleaned_data!!所以这就是给我们写的,让我们覆盖的东西。

  为什么这么说呢?

  我们会发现,点进去类 UserInfo中 forms.Form继承基类方法BaseForm。

  而我们所写的全局钩子中 clean_data也是在基类BaseForm里面。所以代码会先去我们所写的类中找clean()方法,如果没有的话,再继续执行下一步。

  校验成功的话,和局部钩子一样,更新字段值,并返回到clean_data字典里面。但是校验失败的话,抛出的异常和局部钩子有点不一样,他是将异常信息以键值对({\’__all__\’: [value, ]}) 写入errors字典中。全局钩子抛出的异常会添加到__all__里面,所以我们在后台获取异常信息

   后台获取错误信息是这样的:

errors = form.errors.get(\'__all__\')

myforms.errors.get(\'__all__\')[0]

  注意先判断上面的 errors 是否存在,存在的话,在前台获取错误信息如下:

{{ myforms.errors.__all__.0 }}

  

5.6,全局钩子示例:

# 函数名称必须命名为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:
                # 没通过检测,则返回异常信息
                ValidationError(\'两次密码不一致\')
        else:
            # 如果获取的两个变量中,但凡有一个为空,我们就不需要校验了
            return self.cleaned_data

  

5.7,一个完整的forms组件校验

   这里代码主要展示了视图层和form组件和HTML代码。当然前提是我们需要有UserInfo这个数据库。算了还是展示一下models函数吧

  models.py

from django.db import models

# Create your models here.

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)

  

  MyForms.py

from django import forms

from django.forms import widgets
from form_demo.models import UserInfo

from django.core.exceptions import ValidationError

class UserForm(forms.Form):
    name = forms.CharField(min_length=3, label=\'用户名\',
                           error_messages={\'required\': \'用户名最短是3位!!\'},
                           widget=widgets.TextInput(attrs={\'class\': \'form-control\'}))
    pwd = forms.CharField(min_length=3, label=\'密码\',
                          error_messages={\'required\': \'密码最短是3位!!\'},
                          widget=widgets.PasswordInput(attrs={\'class\': \'form-control\'}))
    r_pwd = forms.CharField(min_length=3, label=\'确认密码\',
                            error_messages={\'required\': \'密码最短是3位!!\'},
                            widget=widgets.PasswordInput(attrs={\'class\': \'form-control\'}))
    email = forms.EmailField(label=\'邮箱\',
                             error_messages={\'required\': \'不符合邮箱格式!!\'},
                            widget=widgets.TextInput(attrs={\'class\': \'form-control\'}))
    # 手机号码的规则固定为11位,***
    tel = forms.CharField(min_length=3, label=\'电话\',
                          widget=widgets.TextInput(attrs={\'class\': \'form-control\'}))

    # 函数名称必须以 claen_字段名  的格式
    def clean_name(self):
        # 目的:如果用户名已经注册,则报异常
        val = self.cleaned_data.get(\'name\')

        ret = UserInfo.objects.filter(name=val)
        if not ret:
            return val
        else:
            raise ValidationError(\'用户名已经被注册\')

    def clean_tel(self):
        # 目的:校验手机号码长度为11
        val = self.cleaned_data.get(\'tel\')

        if len(val) == 11:
            return val
        else:
            raise ValidationError("手机号码格式错误")

    # 函数名称必须命名为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:
                # 没通过检测,则返回异常信息
                ValidationError(\'两次密码不一致\')
        else:
            # 如果获取的两个变量中,但凡有一个为空,我们就不需要校验了
            return self.cleaned_data

  

  views.py

from django.shortcuts import render, HttpResponse, redirect

# Create your views here.

from form_demo.Myforms import UserForm


def reg(request):
    if request.method == \'GET\':
        form = UserForm()
        return render(request, \'form/reg.html\', locals())

    elif request.method == \'POST\':
        # form 表单的 name 属性值应该与forms组件字段名称一致
        form = UserForm(request.POST)
        print(form.is_valid())
        # print(forms.clean)
        if form.is_valid():
            print(form.cleaned_data)
            # 校验全部通过,创建数据时,从clean_data中获取数据,
            # 但是必须将其中多于的数据pop掉,如下面代码
            # myform.cleaned_data.pop(\'re_pwd\')
            # models.User.objects.create(**myform.cleaned_data)
            return redirect(\'http://www.baidu.com\')
        else:
            all_error = form.errors.get(\'__all__\')
            if all_error:
                all_error = all_error[0]
            print(form.errors)
            print(form.errors.get(\'__all__\'))
            # return render(request, \'form/reg.html\', locals())


    return render(request, \'form/reg.html\', locals())

  

 reg.html

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
        <!-- 最新版本的 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>

<hr>
<hr>


<div class="container">
    <div class="row">
        <div class="col-md-6 col-lg-offset-3">
            <form action="" method="post">
                {% csrf_token %}
                <p><label>{{ form.name.label }}</label>
                    {{ form.name }}<span class="pull-right error">{{ form.name.errors.0 }}</span>
                </p>
                <p><label>{{ form.pwd.label }}</label>
                    {{ form.pwd }}<span class="pull-right error">{{ form.pwd.errors.0 }}</span>
                </p>
                <p><label>{{ form.r_pwd.label }}</label>
                    {{ form.r_pwd }}<span class="pull-right error">{{ form.r_pwd.errors.0 }}</span><span>{{ all_error }}</span>
                </p>
                <p><label>{{ form.email.label }}</label>:
                    {{ form.email }}<span class="pull-right error">{{ form.email.errors.0 }}</span>
                </p>
                <p><label>{{ form.tel.label }}</label>:
                    {{ form.tel }}<span class="pull-right error">{{ form.tel.errors.0 }}</span>
                </p>
                <p><input type="submit"></p>
            </form>
        </div>

    </div>
</div>



</body>
</html>

  

 我们展示一下效果。

1,我们给数据库存入james这个名字,然后注册james,我们在前端看效果:

2,我们注册电话号码为5位,我们在前端看效果:

3,我们注册密码和确认密码不一致的时候,我们在前端看效果:

 

 

 参考文献:https://www.cnblogs.com/JetpropelledSnake/p/9397889.html#top(局部钩子和全局钩子,本来我放的代码,但是百度找到了这位老铁画的图的,就借花献佛,直接拿来用了,自己再加以详细的解释)

https://www.cnblogs.com/yuanchenqi/articles/7638956.html

版权声明:本文为wj-1314原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/wj-1314/p/10929251.html