专题18:Django之Form,ModelForm

news2025/1/11 2:37:09

原始思路实现添加用户功能的缺点:

1)用户提交的数据没有校验

2)如果用户输入的数据有错误,没有错误提示

3)前端页面上的每一个字段都需要我们重新写一次

4)关联的数据需要手动取获取并循环展示在页面

1、Form

在views.py中:

# MyForm中继承的Form是Django自带的
class MyForm(forms):   
    # widget=forms.Input表示可以在HTML页面中可以显示成input框
    user = forms.CharField(widget=forms.Input)  
    pwd = forms.CharField(widget=forms.Input)
    email = forms.CharField(widget=forms.Input)


def user_add():
    if method.request == "GET":
        form = MyForm()  # 实例化一个MyForm对象
        return render(request, "user_add.html", {"form": form})
    
    

前端user_add.html中:

<form method="post" action="/user/add/">
    {{ form.user }}
    {{ form.pwd }}
    {{ form.email }}
</form>

 上面代码相当于如下代码,两者比较,明显上面使用了form的代码更加简洁

<form method="post" action="/user/add/">
    <input type="text" class="form-control" placeholder="请输入用户名称" name="user">
    <input type="text" class="form-control" placeholder="请输入密码" name="pwd">
    <input type="text" class="form-control" placeholder="请输入邮箱" name="email">
</form>

还有一种比上面更简洁的写法,直接用for循环遍历form

<form method="post" action="/user/add/">
    {% for field in form %}
        {{ field }}
    {% endfor %}
</form>

2、ModelForm(对某些数据库的表做增删改查,推荐用ModelForm)

1)使用ModelForm增加用户

上面的views.py的方法其实也有很麻烦的地方,就是如果我要用字段的话,那我就要在MyForm这个类中把所要用的字段都手动写一遍,如果字段很多的话,就比较麻烦,有没有更加简单的办法呢?

答案是有的,注意看MyForm里面的字段是不是和models.py中类的字段很相似?看下图:

class UserInfo(models.Model):
    """员工表"""
    name = models.CharField(verbose_name="姓名", max_length=16)
    password = models.CharField(verbose_name="密码", max_length=64)
    age = models.IntegerField(verbose_name="年龄")
    account = models.DecimalField(verbose_name="工资", max_digits=10, decimal_places=2, default=0)

    create_time = models.DateField(verbose_name="入职时间")

    gender_choices = (
        (1, "男"),
        (2, "女")
    )
    # gender = models.SmallIntegerField(verbose_name="性别", choices=gender_choices)

    gender = models.SmallIntegerField(verbose_name="性别", choices=gender_choices)

    depart = models.ForeignKey(verbose_name="部门", to="Department", to_field="id",
                               null=True, blank=True, on_delete=models.SET_NULL)

其实,它们之间的内部之间是有关联的,我们如果使用ModelForm的话,可以这样写:

views.py中:

from django import forms

# 注意:如果要用ModelForm,则前面必须写这个类
class MyModelForm(forms.ModelForm):
    class Meta:
        model = models.UserInfo
        # 这里fields这个列表就把models.py中的UserInfo表中的字段关联到了ModelForm这里
        fields = ["name", "password", "age", "account", "create_time", "gender", "depart"]


def user_add():
    if method.request == "GET":
        form = MyModelForm()  # 实例化一个MyModelForm对象
        return render(request, "user_add.html", {"form": form})

同时,ModelForm中的fields列表里的字段不仅可以是数据库的字段,也可以是我们自定义的字段,而新字段相当于新加字段到表中。如下所示:

from django import forms

class MyModelForm(forms.ModelForm):
    xxx = forms.CharField("...")
    class Meta:
        model = UserInfo
        fields = ["name", "password", "age", "account", "create_time", "gender", "depart", "xxx"]

def user_add():
    if method.request == "GET":
        form = MyModelForm()  # 实例化一个MyModelForm对象
        return render(request, "user_add.html", {"form": form})

实操:

views.py中:

from django import forms

class UserModelForm(forms.ModelForm):
    class Meta:
        model = models.UserInfo
        fields = ["name", "password", "age", "account", "create_time", "gender", "depart"]

def user_model_form_add():
    if method.request == "GET":
        form = UserModelForm()  # 实例化一个MyModelForm对象
        return render(request, "user_model_form_add.html", {"form": form})

user_model_form_add.html中:

<form method="post" action="/user/model/form/add/">
    {% csrf_token %}
    {% for field in form %}
        {{ field.label }} :   {{ field }}
    {% endfor %}
</form>

 运行后页面显示如下:

那么问题来了,为什么部门这块显示的不是部门名称,而是一个个Department类的对象呢? 

首先,我们来分析一下models.py中的代码:

from django.db import models

class Department(models.Model):
    """部门表"""
    # title是部门的名字,eg.title=招标部
    title = models.CharField(verbose_name="标题", max_length=32)


class UserInfo(models.Model):
    """员工表"""
    name = models.CharField(verbose_name="姓名", max_length=16)
    password = models.CharField(verbose_name="密码", max_length=64)
    age = models.IntegerField(verbose_name="年龄")
    account = models.DecimalField(verbose_name="工资", max_digits=10, decimal_places=2, default=0)

    create_time = models.DateField(verbose_name="入职时间")

    gender_choices = (
        (1, "男"),
        (2, "女")
    )

    gender = models.SmallIntegerField(verbose_name="性别", choices=gender_choices)

    depart = models.ForeignKey(verbose_name="部门", to="Department", to_field="id",
                               null=True, blank=True, on_delete=models.SET_NULL)

从上面代码中可以看出,models.py中的depart变量其实是从Department表中取到的一连串对象,前端页面显示的正是这一个个Department类的实例化对象。可是我们要让前端页面显示具体的部门名称,所以这里要引入一个知识点:

当我们定义一个类,并且实例化一个该类的对象时,我们可以在类中定义一个__str()__,这样当我们打印这个对象时,输出的就是__str()__中return的值,比如:而这个页面的问题,也可以用这种方法,也就是在Department表中定义一个__str()__方法,return的就是self.title。代码如下:

class Department(models.Model):
    """部门表"""
    # title是部门的名字,eg.title=招标部
    title = models.CharField(verbose_name="标题", max_length=32)
    def __str__(self):
        return self.title

上面的问题解决了,接下来,我们需要给这些输入框增加样式。具体怎么做呢?

 前端user_model_form_add.html中:

{% extends 'layout.html' %}
{% load static %}

{% block css %}
    <link rel="stylesheet" href="{% static 'plugins/bootstrap-datepicker/css/bootstrap-datepicker.min.css' %}">
{% endblock %}

{% block content %}
    <div>
    <div class="container">
        <div class="panel panel-default">
            <div class="panel-heading">
                <h3 class="panel-title">新建用户</h3>
            </div>
            <div class="panel-body">
                <form method="post" action="/user/model/form/add/" novalidate>
                    {% csrf_token %}
                    {% for field in form %}
{#                        框外文字: 框内文字#}
                        <div>{{ field.label }}: {{ field }}</div>
                        <span style="color: red">{{ field.errors.0 }}</span>

                    {% endfor %}
                    <button type="submit" class="btn btn-primary">保 存</button>
                </form>
            </div>
        </div>
    </div>
</div>
{% endblock %}

上面的代码中,样式唯一缺的就是form-control,我们要想办法把这个样式加上。具体时在views.py中通过widgets来加:

from django import forms

class UserModelForm(forms.ModelForm):
    class Meta:
        model = models.UserInfo
        fields = ["name", "password", "age", "account", "create_time", "gender", "depart"]
        widgets = {
            "name": TextInput(attrs={"class": "form-control"}),
            "password": PasswordInput(attrs={"class": "form-control"})
        }

def user_model_form_add():
    if method.request == "GET":
        form = UserModelForm()  # 实例化一个MyModelForm对象
        return render(request, "user_model_form_add.html", {"form": form})

页面如下:

这种方法虽然可以,但是当字段比较多的时候,像这样一个一个的写就会比较麻烦。在我们真正做开发的时候,肯定不会像这样一个一个的写,我们通常这样写:

from django import forms

class UserModelForm(forms.ModelForm):
    class Meta:
        model = models.UserInfo
        fields = ["name", "password", "age", "account", "create_time", "gender", "depart"]
    def __init__(self, *args, **kwargs):
        super().__init__(*args, **kwargs)
        # 循环找到所有插件,添加class="form-control"样式
        for name, field in self.fields.items():
            field.widget.attrs = {"class": "form-control", "place-holder": field.label}

def user_model_form_add():
    if method.request == "GET":
        form = UserModelForm()  # 实例化一个MyModelForm对象
        return render(request, "user_model_form_add.html", {"form": form})

如果我想让某个字段(比如age)不加样式,可以循环中判断:if name == "age": continue

那么如何对用户输入的数据进行校验呢?

from django import forms

class UserModelForm(forms.ModelForm):
    class Meta:
        model = models.UserInfo
        fields = ["name", "password", "age", "account", "create_time", "gender", "depart"]
    def __init__(self, *args, **kwargs):
        super().__init__(*args, **kwargs)
        # 循环找到所有插件,添加class="form-control"样式
        for name, field in self.fields.items():
            field.widget.attrs = {"class": "form-control", "place-holder": field.label}

def user_model_form_add():
    if method.request == "GET":
        form = UserModelForm()  # 实例化一个MyModelForm对象
        return render(request, "user_model_form_add.html", {"form": form})
    elif method.request == "POST":
        form = UserModelForm(data=request.POST)
        # 如果校验成功,可以获取到form.cleaned_data,返回一个字典
        # 如果数据合法,我们将这个新增的用户数据保存到数据库,这里注意:我们不再需要用ORM语句了(虽然用也可以),因为ModelForm知道我们获取用户数据可能会将其保存进数据库,所以我们直接调用form.save()就可以了,数据会直接保存到UserInfo这个表中,因为在上面的ModelForm中我们定义了model = models.UserInfo
        if form.is_valid():
            # print(form.cleaned_data)
            form.save()
            return redirect("/user/list/")
        # 如果校验失败,则可以获取到它所有的错误信息
        else:
            # print(form.errors)
            return render(request, 'user_model_form_add.html', {"form": form})
            

前端页面通过field.errors.0来显示错误信息,field.errors是一个列表,我们只要显示这个列表的第0个元素就行。

{% extends 'layout.html' %}
{% load static %}

{% block css %}
    <link rel="stylesheet" href="{% static 'plugins/bootstrap-datepicker/css/bootstrap-datepicker.min.css' %}">
{% endblock %}

{% block content %}
    <div>
    <div class="container">
        <div class="panel panel-default">
            <div class="panel-heading">
                <h3 class="panel-title">新建用户</h3>
            </div>
            <div class="panel-body">
                <form method="post" action="/user/model/form/add/" novalidate>
                    {% csrf_token %}
                    {% for field in form %}
{#                        框外文字: 框内文字#}
                        <div>{{ field.label }}: {{ field }}</div>
                        <span style="color: red">{{ field.errors.0 }}</span>

                    {% endfor %}
                    <button type="submit" class="btn btn-primary">保 存</button>
                </form>
            </div>
        </div>
    </div>
</div>
{% endblock %}

 以上的代码只能校验输入内容是否为空,那比如,我想规定姓名的长度必须大于3,那么我们需要在ModelForm的类中额外规定一下:name = forms.CharField(min_length=3, label="姓名")

from django import forms

class UserModelForm(forms.ModelForm):

    name = forms.CharField(min_length=3, label="姓名")
    class Meta:
        model = models.UserInfo
        fields = ["name", "password", "age", "account", "create_time", "gender", "depart"]
    def __init__(self, *args, **kwargs):
        super().__init__(*args, **kwargs)
        # 循环找到所有插件,添加class="form-control"样式
        for name, field in self.fields.items():
            field.widget.attrs = {"class": "form-control", "place-holder": field.label}

def user_model_form_add():
    if method.request == "GET":
        form = UserModelForm()  # 实例化一个MyModelForm对象
        return render(request, "user_model_form_add.html", {"form": form})
    elif method.request == "POST":
        form = UserModelForm(data=request.POST)
        # 如果校验成功,可以获取到form.cleaned_data,返回一个字典
        # 如果数据合法,我们将这个新增的用户数据保存到数据库,这里注意:我们不再需要用ORM语句了(虽然用也可以),因为ModelForm知道我们获取用户数据可能会将其保存进数据库,所以我们直接调用form.save()就可以了,数据会直接保存到UserInfo这个表中,因为在上面的ModelForm中我们定义了model = models.UserInfo
        if form.is_valid():
            # print(form.cleaned_data)
            form.save()
            return redirect("/user/list/")
        # 如果校验失败,则可以获取到它所有的错误信息
        else:
            # print(form.errors)
            return render(request, 'user_model_form_add.html', {"form": form})
            

2)使用ModelForm编辑用户信息

{% extends 'layout.html' %}

{% block content %}
    <div>
    <div class="container">
        <div class="panel panel-default">
            <div class="panel-heading">
                <h3 class="panel-title">编辑用户</h3>
            </div>
            <div class="panel-body">
                <form method="post" novalidate>
                    {% csrf_token %}
                    {% for field in form %}
                        <div>{{ field.label }}: {{ field }}</div>
                        <span style="color: red">{{ field.errors.0 }}</span>

                    {% endfor %}
                    <button type="submit" class="btn btn-primary">保 存</button>
                </form>
            </div>
        </div>
    </div>
</div>
{% endblock %}
def user_edit(request, nid):
    """编辑用户"""
    # 根据ID获取到需要编辑的那一行的数据(对象)
    row_object = models.UserInfo.objects.filter(id=nid).first()
    if request.method == 'GET':
        # 在ModelForm中写instance=row_object,则Django会默认将该对象中的每一个值写在输入框中
        form = UserModelForm(instance=row_object)
        return render(request, 'user_edit.html', {"form": form})

    # 把用户post的数据更新到row_object这一行,并赋值给form
    elif request.method == "POST":
        form = UserModelForm(data=request.POST, instance=row_object)
        if form.is_valid():
            form.save()
            return redirect('/user/list/')
        else:
            return render(request, 'user_edit.html', {"form": form})

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/20028.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Effective_Objective-C_1熟悉Objective_C】

文章目录说在前面的熟悉ObjectiveCfirst了解Objective-C的起源1.消息结构和函数调用运行期组件内存管理Objective-C的起源要点总结Second 在类的头文件尽量少饮入其他文件尽量延后引入头文件或者单独开辟一个文件向前声明在类的头文件尽量少饮入其他文件要点总结THIRD-多用字面…

梯度下降法的理解

1 梯度下降法 本文所有的数学定义概念非官方所给&#xff0c;皆来自于个人理解融合 1.1 梯度的定义 个人理解就是能够使函数值增大最快的方向 需要明确的一点&#xff0c;这里说的方向都是自变量变化的方向 1.2 梯度下降法 梯度下降法本质上是用来求解目标函数最小值的一种…

CocosCreater学习1

1.产品定位 Cocos Creator 是以内容创作为核心&#xff0c;实现了脚本化、组件化和数据驱动的游戏开发工具。具备了易于上手的内容生产工作流&#xff0c;以及功能强大的开发者工具套件&#xff0c;可用于实现游戏逻辑和高性能游戏效果。 2.工作流程 3.项目文件夹结构 初次创…

C++入门基础01:指针与引用

C入门基础&#xff1a;指针与引用 指针 #include <iostream> //系统定义头文件一般是尖括号 #include<fstream> #include<string> using namespace std;//引用与指针 //是复合数据类型&#xff0c;与基本数据类型int、double这类是不一样的 //指针存储的是…

Java集合框架详解(一)Collection接口、List接口、ArrayList类、Vector类

一、集合架构图 二、Collection接口 Collection集合的遍历 迭代器&#xff1a;Iterater 方法名称备注Iterater Iterater()返回在此collection的元素上进行迭代的迭代器boolean hasNext()如果有元素可以迭代&#xff0c;返回true&#xff0c;否则返回falseE next()返回迭代的下…

Kafka - Kafka生产者|发送消息|分区策略|提高吞吐量|数据可靠性|数据去重|数据有序

文章目录1. Kafka 生产者2. 生产者发送消息1. 生产者异步发送消息2. 生产者异步发送消息带回调函数3. 生产者同步发送消息3. 生产者发送消息的分区策略1. Kafka 分区好处2. 分区器 DefaultPartitioner 和 ProducerRecord 源码3. 指定 partition 的情况4. 没有指定 partition 但…

从零集成mybatis-plus

Mybatis-Plus特性&#xff1a; 无侵入&#xff0c;即引入它不会对现有工程产生影响。损耗小&#xff0c;启动就会自动注入基本的CRUD&#xff0c;内置通用Mapper、Service等&#xff0c;基本能满足大部分需求。支持主键自动生成&#xff0c;其中包括分布式唯一ID生成器Sequenc…

PTA题目 分段计算居民水费

为鼓励居民节约用水&#xff0c;自来水公司采取按用水量阶梯式计价的办法&#xff0c;居民应交水费y&#xff08;元&#xff09;与月用水量x&#xff08;吨&#xff09;相关&#xff1a;当x不超过15吨时&#xff0c;y4x/3&#xff1b;超过后&#xff0c;y2.5x−17.5。请编写程序…

[附源码]java毕业设计时事资讯平台

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

黑*头条_第1章_项目介绍和工程搭建

黑*头条_第1章_项目介绍和工程搭建 文章目录黑*头条_第1章_项目介绍和工程搭建学习目标1. 项目介绍1.1项目背景1.2 项目概述1.3 项目术语定义2. 项目需求2.1 APP主要功能大纲2.2 APP用例图&#xff08;主要功能&#xff09;2.3 WEMEDIA功能大纲2.4 WEMEDIA用例图&#xff08;主…

STM32嵌入式工程师自我修养

STM32嵌入式工程师自我修养一、STM32必备技能二、程序员必须熟知三、学习STM32自备资料和硬件一、STM32必备技能 1、熟悉 C 语言编程&#xff0c;熟练 STM32CUBEMX,Keil 开发环境。 2、熟悉基于STM32 MCU开发&#xff0c;掌GPIO,TIME,PWM,ADC等外设开发。 3、熟悉USART,IIC,SP…

快速熟悉C++之常用语法

函数重载&#xff08;Overload&#xff09; 规则 函数名相同 参数个数不同、参数类型不同、参数顺序不同 ◼ 注意 返回值类型与函数重载无关 调用函数时&#xff0c;实参的隐式类型转换可能会产生二义性 ◼ 本质 采用了name mangling或者叫name decoration技术 ✓…

Java锁对象

Java锁 1. 对象头 1.1 简介 以32位的 JVM 为例&#xff0c;每个Java对象的对象头都包含了如下信息 # 组成 Mark Word: 锁的信息&#xff0c;hashcode&#xff0c;垃圾回收器标志 Klass Word: 指针&#xff0c;包含当前对象的Class对象的地址&#xff0c;类对象来确…

基于python的校园社团管理系统的设计与实现

摘 要 随着信息技术和网络技术的飞速发展&#xff0c;人类已进入全新信息化时代&#xff0c;传统管理技术已无法高效&#xff0c;便捷地管理信息。为了迎合时代需求&#xff0c;优化管理效率&#xff0c;各种各样的管理系统应运而生&#xff0c;各行各业相继进入信息管理时代&…

JS 类总结

类 class 关键字是 ES6 新增的。类&#xff08;class&#xff09;是ECMAScript 中新的基础性语法糖&#xff0c;本质上还是一个函数&#xff0c;但实际上它使用的仍然是原型和构造函数的概念。并且类受块级作用域限制。 class Person { } console.log(Person);// class Perso…

Java无锁并发

共享资源 1. 不安全场景 package com.nike.erick.d05;import lombok.Getter;import java.util.concurrent.TimeUnit;public class Demo01 {public static void main(String[] args) throws InterruptedException {BankService bankService new BankService();for (int i 0;…

H5 app开启web调试

前言&#xff1a; 在Android app逆向时&#xff0c;H5类型的app的加密通常在js中&#xff0c;所以就需要一种手段来查看源代码&#xff0c;查看加密过程。 0、如何确认h5 app 以狗东为例&#xff1a; 随便选择一个元素&#xff0c;可以看到是控件下的一个类 通过与H5类型的ap…

[附源码]SSM计算机毕业设计在线课程网站JAVA

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

基于java_ssm_vue鲜花在线销售商城网站-计算机毕业设计

现在&#xff0c;许多人都喜欢在节日的时候给家人或朋友送鲜花&#xff0c;但是有时候会因为工作忙而忘记或者是没有时间自己去买&#xff0c;同时也有些人觉得自己去买有些麻烦&#xff0c;所以鲜花网络销售是很有必要的。这个网站应该可以提供提前预定、送货上门等服务。首先…

Arduino开发实例-MAX30100传感器模块连接问题解决

MAX30100传感器模块连接问题解决 MAX30100 是一款集成脉搏血氧饱和度和心率监测传感器解决方案。 它结合了两个 LED、一个光电探测器、优化的光学器件和低噪声模拟信号处理,以检测脉搏血氧饱和度和心率信号。 MAX30100 采用 1.8V 和 3.3V 电源供电,可通过软件关断,待机电流…