zoukankan      html  css  js  c++  java
  • Django——form表单

    1.HTML表单概述

    Django开发的是动态Web服务,而非单纯提供静态页面。动态服务的本质在于和用户进行互动,接收用户的输入,根据输入的不同,返回不同的内容给用户。返回数据是我们服务器后端做的,而接收用户输入就需要靠HTML表单。表单<form>...</form>可以收集其内部标签中的用户输入,然后将数据发送到服务端。

    一个HTML表单必须指定两样东西:

    • 目的地:用户数据发送的目的URL
    • 方式:发送数据所使用的HTTP方法

    例如,Django Admin站点的登录表单包含几个<input>元素:type="text"用于用户名,type="password"用于密码,type="submit"用于“登录"按钮。它还包含一些用户看不到的隐藏的文本字段,Django 使用它们来提高安全性和决定下一步的行为。它还告诉浏览器表单数据应该发往<form>的action属性指定的URL:/admin/,而且应该使用method属性指定的HTTP post方法发送数据。当点击<input type="submit" value="Log in">元素时,数据将发送给/admin/

                                             

    其HTML源码如下:

    <form action="/admin/login/?next=/admin/" method="post" id="login-form">
    
        <input type='hidden' name='csrfmiddlewaretoken'              value='NNHZaDVJGduajNMECXygKZkAt8vyEcw9HS2qm2Vdf7brDZrA0qK1R0I7M2p3TKcs' />
    
        <div class="form-row">
            <label class="required" for="id_username">用户名:</label> 
            <input type="text" name="username" autofocus maxlength="254" required id="id_username" />
        </div>
    
        <div class="form-row">
            <label class="required" for="id_password">密码:</label> 
            <input type="password" name="password" required id="id_password" />
            <input type="hidden" name="next" value="/admin/" />
        </div>
    
        <div class="submit-row">
            <label>&nbsp;</label><input type="submit" value="登录" />
        </div>
    </form>
    View Code

    get和post

    处理表单时候只会用到POST和GET方法。

    GET方法将用户数据以键=值的形式,以‘&’符号组合在一起成为一个整体字符串,最后添加前缀“?”,将字符串拼接到url内,生成一个类似https://docs.djangoproject.com/search/?q=forms&release=1的URL。

    而对于POST方法,浏览器会组合表单数据、对它们进行编码,然后打包将它们发送到服务器,数据不会出现在url中。

    GET方法通常用来请求数据,不适合密码表单这一类保密信息的发送,也不适合数据量大的表单和二进制数据。对于这些类型的数据,应该使用POST方法。但是,GET特别适合网页搜索的表单,因为这种表示一个GET请求的URL可以很容易地设置书签、分享和重新提交。

    2.Django的form表单

    通常情况下,我们需要自己手动在HTML页面中,编写form标签和其内的其它元素。但这费时费力,而且有可能写得不太恰当,数据验证也比较麻烦。有鉴于此,Django在内部集成了一个表单模块,专门帮助我们快速处理表单相关的内容。Django的表单模块给我们提供了下面三个主要功能:

    • 准备和重构数据用于页面渲染
    • 为数据创建HTML表单元素
    • 接收和处理用户从表单发送过来的数据

    编写Django的form表单,非常类似我们在模型系统里编写一个模型。在模型中,一个字段代表数据表的一列,而form表单中的一个字段代表<form>中的一个<input>元素。

    3.表单实例

    3.1基本用法

    以用户注册为例,实现:

    • 提交数据到后台
    • 校验提交数据

    模型:models.py

    class User(models.Model):
        username = models.CharField(max_length=20, unique=True)
        password = models.CharField(max_length=256)
        email = models.EmailField()
        create_time = models.DateTimeField(auto_now_add=True)
    View Code

    模板:register.html

    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>Register</title>
    
    </head>
    <body>
    
    <form action="/register/" method="post">
        {% csrf_token %} 
        <div>
            <label for="user">用户名</label>
            <p><input type="text" name="name" id="user"></p>
        </div>
        <div>
            <label for="pwd">密码</label>
            <p><input type="password" name="pwd" id="pwd"></p>
        </div>
        <div>
            <label for="r_pwd">确认密码</label>
            <p><input type="password" name="r_pwd" id="r_pwd"></p>
        </div>
         <div>
            <label for="email">邮箱</label>
            <p><input type="text" name="email" id="email"></p>
        </div>
        <input type="submit" value="注册">
    </form>
    
    </body>
    </html>
    View Code

    验证组件:form.py

    from django import forms
    from django.forms import widgets
    
    wid_1=widgets.TextInput(attrs={"class":"form-control"})
    wid_2=widgets.PasswordInput(attr={"class":"form-control"})
    
    
    class UserForm(forms.Form):
        name=forms.CharField(max_length=32,widget=wid_1)
        pwd=forms.CharField(max_length=256,widget=wid_2)
        re_pwd=forms.CharField(max_length=256,widget=wid_2)
        email=forms.CharField(widget=wid_1)
    View Code

    视图:views.py

    def register(request):
        if request.method == "POST":
            fm=UserForm(request.POST)    # 接受request.POST参数构造form类的实例
            if fm.is_valid():
                print(fm.cleaned_data)       # 打印所有有效字段(字典)
            else:
                print(fm.errors)       # 打印字典ErrorDict:{"校验错误的字段":["错误信息",]}
                print(fm.errors.get("name"))     # 获取name的错误信息列表ErrorList   ["错误信息",]
            return HttpResponse("OK")
        fm=UserForm()     # 如果是通过GET方法请求数据,返回一个空的表单
        return render(request,"register.html",locals())
    View Code

    3.1模板优化(渲染)

    上例中的模板完全是我们自己写的,我们完全不需要写那么多的html代码,上面的form表单可以写成:

    <form action="/register/" method="post">
        {% csrf_token %} 
        <div>
         {{ fm }}
        </div>
        <input type="submit" value="注册">
    </form>
    View Code

    但最好不要这样写,因为不太方便渲染,就是太丑的意思。

    除了{{ form }}模板语言,简单地将表单渲染到HTML页面中了,实际上,有更多的方式:

    • {{ form.as_table }} 将表单渲染成一个表格元素,每个输入框作为一个<tr>标签
    • {{ form.as_p }} 将表单的每个输入框包裹在一个<p>标签内 tags
    • {{ form.as_ul }} 将表单渲染成一个列表元素,每个输入框作为一个<li>标签

    注意:你要自己手动编写<table><ul>标签。

    用法是一样的,比如渲染个表格

    <form action="/register/" method="post">
        {% csrf_token %} 
        <div>
            <table>{{ fm.as_table }}</table>
        </div>
        <input type="submit" value="注册">
    </form>    
    View Code

    3.1手动渲染

    直接{{ form }}虽然好,啥都不用操心,但是往往并不是你想要的,比如你要使用CSS和JS,比如你要引入Bootstarps框架,这些都需要对表单内的input元素进行额外控制,那怎么办呢?手动渲染字段就可以了。

    也就是把{{ form }}拆开,这样就可以对每个字段进行你想要的渲染。

    <form action="/register/" method="post">
        <label for="{{ fm.name.id_for_label }}">用户名:</label>
        {{ fm.name }}
        <label for="{{ fm.pwd.id_for_label }}">密码:</label>
        {{ fm.pwd }}
        <label for="{{ fm.re_pwd.id_for_label }}">确认密码:</label>
        {{ fm.re_pwd }}
        {{ form.cc_myself.errors }}
        <label for="{{ fm.email.id_for_label }}">邮箱:</label>
        {{ fm.email }}
    <input type="submit" value="注册">
    </form>
    View Code

    其中的label标签甚至可以用label_tag()方法来生成,需要在form组件添加label,于是可以简写成下面的样子:

    <form action="/register/" method="post">
        {{ fm.name.label_tag }}
        {{ fm.name }}
        ....
    <input type="submit" value="注册">
    </form>    
    View Code

    4.显示错误与循环表单字段

    如果验证错误,则需要让用户看到错误信息

    视图:views.py

    def register(request):
        if request.method == "POST":
            fm=UserForm(request.POST)    
            if fm.is_valid():
                print(fm.cleaned_data)       
            else:
                print(fm.errors)       # 打印字典ErrorDict:{"校验错误的字段":["错误信息",]}
                print(fm.errors.get("name"))     # 获取name的错误信息列表ErrorList   ["错误信息",]
            return  render(request,"register.html",locals())     # 将包括错误信息的表单返回
        fm=UserForm()     
        return render(request,"register.html",locals())
    View Code

    模板:register.html

    <form action="/register/" method="post" novalidate>
        {% csrf_token %}
    
        {% for field in fm %}     # 循环表单字段
            <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" value="注册">
    
    </form>
    View Code

    5.局部钩子与全局钩子

    form组件预留了一些钩子函数,方便定制一些错误信息

    from django import forms
    from django.forms import widgets
    
    wid_1=widgets.TextInput(attrs={"class":"form-control"})
    wid_2=widgets.PasswordInput(attr={"class":"form-control"})
    
    
    class UserForm(forms.Form):
        name=forms.CharField(max_length=32,widget=wid_1)
        pwd=forms.CharField(max_length=256,widget=wid_2)
        re_pwd=forms.CharField(max_length=256,widget=wid_2)
        email=forms.CharField(widget=wid_1)
    
        # 局部钩子
        def clean_name(self):
            username=self.cleaned_data.get("name")
            if username.isdigit()
                raise ValidationError("用户名不能是纯数字!")
            else:
                 return username
    
    
        # 全局钩子
        def clean(self):
            def clean(self):
            pwd=self.cleaned_data.get("pwd")
            r_pwd=self.cleaned_data.get("r_pwd")
    
            if pwd==r_pwd:
                return self.cleaned_data
            else:
                raise ValidationError('两次密码不一致!')
    View Code

    模板

    <form action="/register/" method="post" novalidate>
        {% csrf_token %}
    
        {% for field in fm %}     # 循环表单字段
            <div>
                <label for="">{{ field.label }}</label>
                {{ field }} <span class="pull-right" style="color: red">
                {% if field.name == 're_pwd' %}
                    <span>{{ clean_error.0 }}</span>     # 全局钩子的错误信息
                {% endif %}
                {{ field.errors.0 }}
                </span>   # 显示错误
            </div>
        {% endfor %}
        <input type="submit" class="btn b
    View Code
    终日不为以思,无益,不如学也
  • 相关阅读:
    「洛谷 NOIP 计划 2021」【学习1】降维技巧
    组合数取模 合集
    浅谈并查集
    四边形不等式优化 dp (doing)
    qbxt数学五一Day4
    qbxt五一数学Day3
    qbxt五一数学Day2
    qbxt五一数学Day1
    浅谈拉格朗日插值
    10-交换排序:冒泡排序
  • 原文地址:https://www.cnblogs.com/lymlike/p/11556045.html
Copyright © 2011-2022 走看看