Python的Django框架中forms表單類(lèi)的使用方法詳解
Form表單的功能
- 自動(dòng)生成HTML表單元素
- 檢查表單數(shù)據(jù)的合法性
- 如果驗(yàn)證錯(cuò)誤,重新顯示表單(數(shù)據(jù)不會(huì)重置)
- 數(shù)據(jù)類(lèi)型轉(zhuǎn)換(字符類(lèi)型的數(shù)據(jù)轉(zhuǎn)換成相應(yīng)的Python類(lèi)型)
Form相關(guān)的對(duì)象包括
- Widget:用來(lái)渲染成HTML元素的工具,如:forms.Textarea對(duì)應(yīng)HTML中的<textarea>標(biāo)簽
- Field:Form對(duì)象中的一個(gè)字段,如:EmailField表示email字段,如果這個(gè)字段不是有效的email格式,就會(huì)產(chǎn)生錯(cuò)誤。
- Form:一系列Field對(duì)象的集合,負(fù)責(zé)驗(yàn)證和顯示HTML元素
- Form Media:用來(lái)渲染表單的CSS和JavaScript資源。
Form Objects
Form對(duì)象封裝了一系列Field和驗(yàn)證規(guī)則,F(xiàn)orm類(lèi)都必須直接或間接繼承自django.forms.Form,定義Form有兩種方式:
方法一:直接繼承Form
from django import forms class ContactForm(forms.Form): subject = forms.CharField(max_length=100,label='主題') message = form.CharField(widget=forms.TextArea) sender = form.EmailField() cc_myself = forms.BooleanField(required=False)
方法二:結(jié)合Model,繼承django.forms.ModelForm
#models.py class Contact(models.Model): title = models.CharField(max_length=30) content = models.CharField(max_length=20) #form.py class ConotactForm(ModelForm): class Meta: model = Contact field = ('title','content') #只顯示model中指定的字段
在視圖(view)中使用form
在view函數(shù)中使用form的一般情景是:
view.py:
form django.shortcuts import render form django.http import HttpResponseRedirect def contact(request): if request.method=="POST": form = ContactForm(request.POST) if form.is_valid(): #所有驗(yàn)證都通過(guò) #do something處理業(yè)務(wù) return HttpResponseRedirect('/') else: form = ContactForm() return render(request,'contact.html',{'form':form})
contact.html:
<form action='/contact/' method='POST'> {% for field in form %} <div class = 'fieldWrapper'> {{field.label_tag}}:{{field}} {{field.errors}} </div> {% endfor %} <div class='fieldWrapper'> <p><input type='submit' value='留言'></p></div> </form>
處理表單數(shù)據(jù)
form.is_valid()返回true后,表單數(shù)據(jù)都被存儲(chǔ)在form.cleaned_data對(duì)象中(字典類(lèi)型,意為經(jīng)過(guò)清洗的數(shù)據(jù)),而且數(shù)據(jù)會(huì)被自動(dòng)轉(zhuǎn)換為Python對(duì)象,如:在form中定義了DateTimeField,那么該字段將被轉(zhuǎn)換為datetime類(lèi)型,還有諸如:IntegerField、FloatField
if form.is_valid(): subject = form.cleaned_data['subject'] message = form.cleaned_data['message'] sender = form.cleaned_data['sender'] cc_myself = form.cleaned_data['cc_myself'] recipients = ['info@example.com'] if cc_myself: recipients.append(sender) from django.core.mail import send_mail send_mail(subject, message, sender, recipients) return HttpResponseRedirect('/thanks/') # Redirect after POST
Form的簡(jiǎn)單使用方法就這些。 另:
在模版中顯示表單的幾種方式:
顯示form找template中的方法多種多樣,也可以自定義:
<form action="/contact/" method="post">{% csrf_token %} {{ form.as_p }} <input type="submit" value="Submit" /> </form>
還可以使用form.as_table、form.as_ul,分別表示用<p>標(biāo)簽,<table>標(biāo)簽和<ul>表示顯示表單。如果要自定義,你只要獲取到每個(gè)元素的值就行:
<form action="/contact/" method="post"> {{ form.non_field_errors }} <div class="fieldWrapper"> {{ form.subject.errors }} <label for="id_subject">Email subject:</label> {{ form.subject }} </div> <div class="fieldWrapper"> {{ form.message.errors }} <label for="id_message">Your message:</label> {{ form.message }} </div> <div class="fieldWrapper"> {{ form.sender.errors }} <label for="id_sender">Your email address:</label> {{ form.sender }} </div> <div class="fieldWrapper"> {{ form.cc_myself.errors }} <label for="id_cc_myself">CC yourself?</label> {{ form.cc_myself }} </div> <p><input type="submit" value="Send message" /></p> </form>
每個(gè)form字段都可以使用 {{form.name_of_field}}得到。
也可以通過(guò)迭代form,每個(gè)迭代元素對(duì)應(yīng)的是form里面的field
<form action="/contact/" method="post"> {% for field in form %} <div class="fieldWrapper"> {{ field.errors }} {{ field.label_tag }}: {{ field }} </div> {% endfor %} <p><input type="submit" value="Send message" /></p> </form>
{{field}}有如下屬性:
{{field.lable}},如:Email address {{field.label_tag}},如: <label for=id_email>Email address</label> {{field.value}} 如:someone.@gmail.com {{field.errors}}
實(shí)例:構(gòu)建表單
第一步:首先在models.py中定義一個(gè)表單模型
class RemarkForm(forms.Form): subject = forms.CharField(max_length=100 ,label='留言標(biāo)題') mail = forms.EmailField(label='電子郵件') topic = forms.ChoiceField(choices=TOPIC_CHOICES,label='選擇評(píng)分') message = forms.CharField(label='留言?xún)?nèi)容',widget=forms.Textarea) cc_myself = forms.BooleanField(required=False ,label='訂閱該貼')
那個(gè)topic中的choices需要在models.py中定義一個(gè)數(shù)組.
TOPIC_CHOICES = ( ('leve1', '差評(píng)'), ('leve2', '中評(píng)'), ('leve3', '好評(píng)'), )
這樣,之后在html顯示的表單就采用這個(gè)模型的數(shù)據(jù)了。
另外還有一種定義表單模型的方式,那就是直接繼承另一個(gè)models。如果我們?cè)趍odels里設(shè)計(jì)數(shù)據(jù)庫(kù)時(shí),已經(jīng)設(shè)計(jì)好了一個(gè)類(lèi)(就是數(shù)據(jù)庫(kù)的表)之后想復(fù)用這個(gè)類(lèi)的信息來(lái)作為表單的模型,那么很簡(jiǎn)單,同樣是在models中的一個(gè)類(lèi)
class Advertisement(models.Model): #訂單編號(hào) OrderID = models.ForeignKey(OrderInfo) #//廣告標(biāo)題# Title = models.CharField(max_length = 36) #//廣告內(nèi)容# Content = models.CharField(max_length = 600)
注意,他的類(lèi)型是models.Model,是做數(shù)據(jù)庫(kù)ORM用的。
然后怎么關(guān)聯(lián)呢?
需要導(dǎo)入一個(gè)新的類(lèi) (ModelForm)
from django.forms import ModelForm class ContactForm(ModelForm): class Meta: model = Advertisement fields = ('Title', 'CustomerID')
這里的Advertisement就是之前那個(gè)ORM的模型。
第二步:OK,繼續(xù)我們的表單,下一步需要做什么呢?開(kāi)始在views.py里寫(xiě)對(duì)表單的調(diào)用吧.
def remark(request):
if request.method == 'POST': # 如果表單被提交 form = ContactForm(request.POST) # 獲取Post表單數(shù)據(jù) if form.is_valid(): # 驗(yàn)證表單 return HttpResponseRedirect('/') # 跳轉(zhuǎn) else: form = ContactForm() #獲得表單對(duì)象 return render_to_response('message.html', { 'form': form, })
整個(gè)代碼很簡(jiǎn)單,我就不過(guò)多的解釋了。
第三步:我們知道django里任何訪問(wèn)都是通過(guò)urls.py來(lái)管理的。所以下面我們需要配置一個(gè)路徑。
(r'^message', 'iring.views.remark'),
第四步:最后應(yīng)該是建立一個(gè)模板的時(shí)候了,因?yàn)槲覀冏罱K是輸出到html上的.注意views的remark函數(shù)最后一行
return render_to_response('message.html', { 'form': form, })
就是講當(dāng)前的表單對(duì)象輸出到message.html里,自動(dòng)生成一個(gè)表單。
所以,先構(gòu)建一個(gè)html.
這個(gè)html模板很簡(jiǎn)單,除去不必要的CSS,我只給出核心部分好啦。
<form action="/message/" method="POST"> {% for field in form %} <div class="fieldWrapper"> {{ field.label_tag }}:{{ field }} {{ field.errors }} div> {% endfor %} <div class="fieldWrapper"><p><input type="submit" value="留言" />p>div> form>
{% for field in form %} {% endfor %}
用于對(duì)表單對(duì)象里的元素進(jìn)行遍歷,然后通過(guò)
{{ field.label_tag }} {{ field }} {{ field.errors }}
這三個(gè)標(biāo)簽來(lái)輸出,注意{{ field.errors }}默認(rèn)情況下是不會(huì)輸出的,只有當(dāng)驗(yàn)證表單正確性時(shí)才會(huì)輸出內(nèi)容。
最終我們通過(guò):
http://youdjangourl/message來(lái)訪問(wèn)這個(gè)例子
相關(guān)文章
tkinter高級(jí)布局之PanedWindow和notebook詳解
本文主要介紹了tkinter中的兩種布局控件,分別是可以動(dòng)態(tài)劃分子控件的PanedWindow,和提供了選項(xiàng)卡工具的notebook,感興趣的小伙伴可以學(xué)習(xí)一下2023-08-08淺析python打包工具distutils、setuptools
python包在開(kāi)發(fā)中十分常見(jiàn),一般的使用套路是所有的功能做一個(gè)python模塊包,打包模塊,然后發(fā)布,安裝使用。這篇文章給大家介紹了python打包工具distutils、setuptools的相關(guān)知識(shí),感興趣的朋友一起看看吧2018-04-04使用TensorFlow搭建一個(gè)全連接神經(jīng)網(wǎng)絡(luò)教程
今天小編就為大家分享一篇使用TensorFlow搭建一個(gè)全連接神經(jīng)網(wǎng)絡(luò)教程,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-02-02Python爬蟲(chóng)Scrapy框架IP代理的配置與調(diào)試
在調(diào)試爬蟲(chóng)的時(shí)候,新手都會(huì)遇到關(guān)于ip的錯(cuò)誤,本文就來(lái)介紹一下Python爬蟲(chóng)Scrapy框架IP代理的配置與調(diào)試,具有一定的參考價(jià)值,感興趣的可以了解一下2021-12-12簡(jiǎn)單談?wù)凱ython中的模塊導(dǎo)入
Python 模塊(Module),是一個(gè) Python 文件,以 .py 結(jié)尾,包含了 Python對(duì)象定義和Python語(yǔ)句,下面這篇文章主要給大家介紹了關(guān)于Python模塊導(dǎo)入的相關(guān)資料,需要的朋友可以參考下2021-10-10Python統(tǒng)計(jì)python文件中代碼,注釋及空白對(duì)應(yīng)的行數(shù)示例【測(cè)試可用】
這篇文章主要介紹了Python統(tǒng)計(jì)python文件中代碼,注釋及空白對(duì)應(yīng)的行數(shù),涉及Python針對(duì)py文件的讀取、遍歷、判斷、統(tǒng)計(jì)等相關(guān)操作技巧,需要的朋友可以參考下2018-07-07