Django學習筆記(14)——AJAX與Form組件知識補充(局部鈎子和全局鈎子詳解)


   我在之前做了一個關於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


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM