首页 > 其他分享 >【20.6】Django框架Form组件之模型表单ModelForm

【20.6】Django框架Form组件之模型表单ModelForm

时间:2024-03-18 23:36:25浏览次数:28  
标签:save Form 20.6 模型 表单 Meta Django ModelForm class

【一】引入

  • 如果你正在构建一个数据库驱动的应用,那么你可能会有与Django的模型紧密映射的表单。
  • 比如,你有个BlogComment模型,并且你还想创建一个表单让大家提交评论到这个模型中。
  • 在这种情况下,写一个forms.Form类,然后在表单类中定义字段,这种一般创建表单的做法是冗余的,因为你已经在ORM模型model中定义了字段的属性和功能,完全没必要重新写一遍字段。

【二】核心用法

  • 基于这个原因,Django提供一个辅助类帮助我们利用Django的ORM模型model创建Form。
from django.forms import ModelForm
from myapp.models import Article


# 创建表单类
class ArticleForm(ModelForm):
    class Meta:
        model = Article
        fields = ['pub_date', 'headline', 'content', 'reporter']


# 创建一个表单,用于添加文章
form = ArticleForm()

# 创建表单修改已有的文章
article = Article.objects.get(pk=1)
form = ArticleForm(instance=article)

  • 用法的核心是:

    • 首先从django.forms导入ModelForm;

    • 编写一个自己的类,继承ModelForm;

    • 在新类里,设置元类Meta;

    • 在Meta中,设置model属性为你要关联的ORM模型,这里是Article;

    • 在Meta中,设置fields属性为你要在表单中使用的字段列表;

    • 列表里的值,应该是ORM模型model中的字段名。

  • 上面的例子中,因为model和form比较简单,字段数量少,看不出这么做的威力和效率。

  • 但如果是那种大型项目,每个模型的字段数量几十上百,这么做的收益将非常巨大,而且后面还有一招提高效率的大杀器,也就是一步保存数据的操作。

【三】字段类型

  • 生成的Form类中将具有和指定的模型字段对应的表单字段,顺序为fields属性列表中指定的顺序。
  • 每个模型字段有一个对应的默认表单字段。
  • 比如,模型中的CharField表现成表单中的CharField。
  • 模型中的ManyToManyField字段会表现成MultipleChoiceField字段。下面是完整的映射列表:
模型字段 表单字段
AutoField 在Form类中无法使用
BigAutoField 在Form类中无法使用
BigIntegerField IntegerField,最小-9223372036854775808,最大9223372036854775807.
BooleanField BooleanField
CharField CharField,同样的最大长度限制。如果model设置了null=True,Form将使用empty_value
CommaSeparatedIntegerField CharField
DateField DateField
DateTimeField DateTimeField
DecimalField DecimalField
EmailField EmailField
FileField FileField
FilePathField FilePathField
FloatField FloatField
ForeignKey ModelChoiceField
ImageField ImageField
IntegerField IntegerField
IPAddressField IPAddressField
GenericIPAddressField GenericIPAddressField
ManyToManyField ModelMultipleChoiceField
NullBooleanField NullBooleanField
PositiveIntegerField IntegerField
PositiveSmallIntegerField IntegerField
SlugField SlugField
SmallIntegerField IntegerField
TextField CharField,并带有widget=forms.Textarea参数
TimeField TimeField
URLField URLField
  • 可以看出,Django在设计model字段和表单字段时存在大量的相似和重复之处。

  • ManyToManyField和 ForeignKey字段类型属于特殊情况:

    • ForeignKey被映射成为表单类的django.forms.ModelChoiceField,它的选项是一个模型的QuerySet,也就是可以选择的对象的列表,但是只能选择一个。

    • ManyToManyField被映射成为表单类的django.forms.ModelMultipleChoiceField,它的选项也是一个模型的QuerySet,也就是可以选择的对象的列表,但是可以同时选择多个,多对多嘛。

  • 同时,在表单属性设置上,还有下面的映射关系:

    • 如果模型字段设置blank=True,那么表单字段的required设置为False。 否则,required=True。

    • 表单字段的label属性根据模型字段的verbose_name属性设置,并将第一个字母大写。

    • 如果模型的某个字段设置了editable=False属性,那么它表单类中将不会出现该字段。道理很简单,都不能编辑了,还放在表单里提交什么?

    • 表单字段的help_text设置为模型字段的help_text

    • 如果模型字段设置了choices参数,那么表单字段的widget属性将设置成Select框,其选项来自模型字段的choices。选单中通常会包含一个空选项,并且作为默认选择。如果该字段是必选的,它会强制用户选择一个选项。 如果模型字段具有default参数,则不会添加空选项到选单中。

【四】完整示例

【1】模型

from django.db import models
from django.forms import ModelForm

TITLE_CHOICES = (
    ('MR', 'Mr.'),
    ('MRS', 'Mrs.'),
    ('MS', 'Ms.'),
)

class Author(models.Model):
    name = models.CharField(max_length=100)
    title = models.CharField(max_length=3, choices=TITLE_CHOICES)
    birth_date = models.DateField(blank=True, null=True)

    def __str__(self):              # __unicode__ on Python 2
        return self.name

class Book(models.Model):
    name = models.CharField(max_length=100)
    authors = models.ManyToManyField(Author)

class AuthorForm(ModelForm):
    class Meta:
        model = Author
        fields = ['name', 'title', 'birth_date']

class BookForm(ModelForm):
    class Meta:
        model = Book
        fields = ['name', 'authors']
  • 上面的ModelForm子类基本等同于下面的定义方式(唯一的区别是save()方法):
from django import forms

class AuthorForm(forms.Form):
    name = forms.CharField(max_length=100)
    title = forms.CharField(
        max_length=3,
        widget=forms.Select(choices=TITLE_CHOICES),
    )
    birth_date = forms.DateField(required=False)

class BookForm(forms.Form):
    name = forms.CharField(max_length=100)
    authors = forms.ModelMultipleChoiceField(queryset=Author.objects.all())

【五】ModelForm的验证

【1】验证ModelForm

  • 验证ModelForm主要分两步

  • 表单本身的验证

  • 验证模型实例

【2】解释

  • 与普通的表单验证类似,模型表单的验证也是调用is_valid()方法或访问errors属性时隐式触发,在调用 full_clean() 时显式触发。

  • 模型的验证(Model.full_clean())紧跟在表单的clean()方法调用之后。

  • 通常情况下,我们使用Django内置的验证器就好了。如果需要,可以重写模型表单的clean()来提供额外的验证,方法和普通的表单一样。

  • 作为验证过程的一部分, ModelForm 将调用模型上与表单字段对应的每个字段的 clean() 方法,模型的整体clean()方法和最后的唯一性验证方法。这部分内容在前面的验证器章节有详细介绍。

  • 表单字段 级别或者 表单 Meta 级别定义的错误信息优先级总是高于在 模型字段 级别定义的。

  • 模型字段 上定义的错误信息只有在 模型验证步骤引发 ValidationError 时才会使用,并且没有在表单级定义相应的错误信息。

  • 你可以添加 NON_FIELD_ERRORS 键到 ModelForm 内部的 Meta 类的 error_messages 中,来显示模型验证引发的 NON_FIELD_ERRORS 错误信息。如下所示:

from django.core.exceptions import NON_FIELD_ERRORS
from django.forms import ModelForm

class ArticleForm(ModelForm):
    class Meta:
        error_messages = {
            NON_FIELD_ERRORS: {
                'unique_together': "%(model_name)s's %(field_labels)s are not unique.",
            }
        }

【六】save() 方法

  • 每个 ModelForm 也有 save() 方法。
  • 此方法根据绑定到表单的数据创建并保存数据库对象。
  • ModelForm 的子类可接受一个现有的模型实例作为关键字参数 instance ;如果提供了,则 save() 会更新这个实例。
  • 如果没有,则 save() 会创建一个对应模型的新实例。
from myapp.models import Article
from myapp.forms import ArticleForm

# Create a form instance from POST data.
f = ArticleForm(request.POST)

# Save a new Article object from the form's data.
new_article = f.save()

# Create a form to edit an existing Article, but use
# POST data to populate the form.
a = Article.objects.get(pk=1)
f = ArticleForm(request.POST, instance=a)
f.save()
  • 请注意,如果表单 尚未验证 ,调用 save() 将通过检查 form.errors 来实现验证。如果表单验证不过,则会引发 ValueError —— 比如,如果 form.errors 返回 True
  • save() 方法接受一个可选参数 commit ,它的值是 True 或者 False
    • 如果调用 save() 的时候使用 commit=False ,那么它会返回一个尚未保存到数据库的对象。
    • 在这种情况下,需要您自己在生成的模型实例上调用 save()
    • 如果要在保存对象之前对对象执行自定义操作,或者要使用一个专用的模型保存选项 ,这很有用。
    • commit 的值默认为 True
  • 如果你的模型具有多对多关系,并且在保存表单时指定了 commit=False ,Django无法立即保存多对多关系的表单数据。这是因为实例的多对多数据只有实例在数据库中存在时才能保存。
  • 要解决这个问题,Django会在每次使用 commit=False 保存表单时,向 ModelForm 子类添加一个 save_m2m() 方法。在您手动保存表单生成的实例后,可以调用 save_m2m() 来保存多对多的表单数据。例如:
# Create a form instance with POST data.
f = AuthorForm(request.POST)

# Create, but don't save the new author instance.
new_author = f.save(commit=False)

# Modify the author in some way.
new_author.some_field = 'some_value'

# Save the new instance.
new_author.save()

# Now, save the many-to-many data for the form.
f.save_m2m()

只有在使用 save(commit=False) 的时候才需要调用 save_m2m()

当你在表单上使用 save() 时,无需调用其他方法,所有数据(包括多对多数据)都会被保存。例如:

# Create a form instance with POST data.
a = Author()
f = AuthorForm(request.POST, instance=a)

# Create and save the new author instance. There's no need to do anything else.
new_author = f.save()
  • 除了 save()save_m2m() 方法之外,ModelForm 与普通的表单工作方式一样。
    • 例如,用 is_valid() 方法来检查合法性
    • is_multipart() 方法来确定表单是否需要multipart文件上传(之后是否必须将 request.FILES 传递给表单)
    • 等等。

【七】ModelForm的字段选择

  • 强烈建议使用ModelForm的fields属性,在赋值的列表内,一个一个将要使用的字段添加进去。这样做的好处是,安全可靠。
  • 然而,有时候,字段太多,或者我们想偷懒,不愿意一个一个输入,也有简单的方法:

【1】__all__:

  • 将fields属性的值设为__all__,表示将映射的模型中的全部字段都添加到表单类中来。
from django.forms import ModelForm

class AuthorForm(ModelForm):
    class Meta:
        model = Author
        fields = '__all__'

【2】exclude属性:

  • 表示将model中,除了exclude属性中列出的字段之外的所有字段,添加到表单类中作为表单字段。
class PartialAuthorForm(ModelForm):
    class Meta:
        model = Author
        exclude = ['title']
  • 因为Author模型有3个字段name、birth_date和title,上面的例子会让birth_date和name出现在表单中。
  • 另外,如果你在某个模型字段中定义了 editable=False , 则任何使用 ModelForm 给该模型创建的表单都不会包含这个字段。
  • 任何没在上面逻辑中包含的表单字段都会不被表单的 save() 方法处理。另外,如果手动将排除的字段添加回表单,它们也不会被模型实例初始化。

【八】重新定义ModelForm字段

  • 在前面,我们有个表格,展示了从模型到模型表单在字段上的映射关系。
  • 通常,这是没有什么问题,直接使用,按默认的来就行了。
  • 但是,有时候可能这种默认映射关系不是我们想要的,或者想进行一些更加灵活的定制,那怎么办呢?

使用Meta类内部的widgets属性!

  • widgets属性接收一个数据字典。
    • 其中每个元素的键必须是模型中的字段名之一
    • 键值就是我们要自定义的内容了,具体格式和写法,参考下面的例子。
  • 例如,如果你想要让Author模型中的name字段的类型从CharField更改为<textarea>,而不是默认的<input type="text">,可以如下重写字段的Widget:
from django.forms import ModelForm, Textarea
from myapp.models import Author

class AuthorForm(ModelForm):
    class Meta:
        model = Author
        fields = ('name', 'title', 'birth_date')
        widgets = {
            'name': Textarea(attrs={'cols': 80, 'rows': 20}), # 关键是这一行
        }
  • 上面还展示了添加样式参数的格式。
  • 如果你希望进一步自定义字段,还可以指定Meta类内部的error_messageshelp_textslabels属性,比如:
from django.utils.translation import ugettext_lazy as _

class AuthorForm(ModelForm):
    class Meta:
        model = Author
        fields = ('name', 'title', 'birth_date')
        labels = {
            'name': _('Writer'),
        }
        help_texts = {
            'name': _('Some useful help text.'),
        }
        error_messages = {
            'name': {
                'max_length': _("This writer's name is too long."),
            },
        }
  • 还可以指定field_classes属性将字段类型设置为你自己写的表单字段类型。
  • 例如,如果你想为slug字段使用MySlugFormField,可以像下面这样:
from django.forms import ModelForm
from myapp.models import Article

class ArticleForm(ModelForm):
    class Meta:
        model = Article
        fields = ['pub_date', 'headline', 'content', 'reporter', 'slug']
        field_classes = {
            'slug': MySlugFormField,
        }
  • 最后,如果你想完全控制一个字段,包括它的类型,验证器,是否必填等等。
  • 可以显式地声明或指定这些性质,就像在普通表单中一样。
  • 比如,如果想要指定某个字段的验证器,可以显式定义字段并设置它的validators参数:
from django.forms import ModelForm, CharField
from myapp.models import Article

class ArticleForm(ModelForm):
    slug = CharField(validators=[validate_slug])

    class Meta:
        model = Article
        fields = ['pub_date', 'headline', 'content', 'reporter', 'slug']

【九】启用字段本地化

  • 默认情况下,ModelForm中的字段不会本地化它们的数据。
  • 可以使用Meta类的localized_fields属性来启用字段的本地化功能。
from django.forms import ModelForm
from myapp.models import Author
class AuthorForm(ModelForm):
    class Meta:
        model = Author
        localized_fields = ('birth_date',)
  • 如果localized_fields设置为__all__这个特殊的值,所有的字段都将本地化。

【十】表单的继承

  • ModelForms是可以被继承的。
  • 子模型表单可以添加额外的方法和属性,比如下面的例子:
class EnhancedArticleForm(ArticleForm):
    def clean_pub_date(self):
        ...
  • 以上创建了一个ArticleForm的子类EnhancedArticleForm,并增加了一个clean_pub_date方法。
  • 还可以修改Meta.fieldsMeta.exclude列表,只要继承父类的Meta类,如下所示:
class RestrictedArticleForm(EnhancedArticleForm):
    class Meta(ArticleForm.Meta):
        exclude = ('body',)

【十一】提供初始值

  • 可以在实例化一个表单时通过指定initial参数来提供表单中数据的初始值。
  • 以这种方式提供的初始值会覆盖表单字段的初始值以及对应模型实例的初始值。
  • 例如:
article = Article.objects.get(pk=1)
article.headline
# 'My headline'

form = ArticleForm(initial={'headline': 'Initial headline'}, instance=article)
form['headline'].value()
# 'Initial headline'

标签:save,Form,20.6,模型,表单,Meta,Django,ModelForm,class
From: https://www.cnblogs.com/dream-ze/p/18081782

相关文章

  • 【20.5】Django框架Form组件之源码
    【一】切入点切入点form_obj.is_valid()defis_valid(self):"""ReturnTrueiftheformhasnoerrors,orFalseotherwise."""returnself.is_boundandnotself.errors如果is_valid要想返回True那么self.is_bound要为Trueself.errors......
  • Django 用re_path()方法正则匹配复杂路由
    app1.url.pyfromdjango.urlsimportpath,re_pathfrom.importviewsurlpatterns=[path('index',views.index,name='index'),path('test',views.test,name='test'),path('test_int/<int:id>/......
  • 【20.0】Django框架之Form组件
    【一】Django的表单系统【1】使用表单(1)原始表单我们之前在HTML页面中利用form表单向后端提交数据时都会写一些获取用户输入的标签并且用form标签把它们包起来。与此同时我们在好多场景下都需要对用户的输入做校验比如校验用户是否输入输入的长度和格式等正不正确。......
  • 【10.0】Django框架之路由层
    【零】Django的路由系统【1】什么是URL配置(URLconf)URL调度器|Django文档|Django(djangoproject.com)URL配置(URLconf)就像Django所支撑网站的目录。它的本质是URL与要为该URL调用的视图函数之间的映射表。你就是以这种方式告诉Django,对于这个URL调用这段代......
  • 【9.0】Django框架之生命周期流程图
    【5.0】Django框架之请求生命周期流程图解释图:浏览器>>发送请求(Http请求)>>web服务网关接口(django默认的wsgiref模块不能承受高并发,最大只有1000左右)>>中间件>>缓存数据库(返回给中间件已经缓存过的数据)>>urls.py(路由层)>>views.py(视图层)>>templates(模板......
  • 【13.0】Django框架之模型层
    【一】前言Django自带的sqlite3数据对日期格式不敏感,处理的时候容易出错【1】测试脚本测试脚本当我们只想要测试Django中的某一个py文件的内容时,我们可以不需要书写前后端交互的形式,而是直接写一个测试脚本即可测试环境的准备在测试文件中书写下面内容这内容其......
  • 【12.0】Django框架之模板层
    【零】Django模板系统模板|Django文档|Django(djangoproject.com)【一】模板语法的传值{{}}:变量相关{%%}:逻辑相关【0】变量(1)传值在Django的模板语言中按此语法使用:{{变量名}}。当模版引擎遇到一个变量,它将计算这个变量,然后用结果替换掉它本身。变......
  • 【11.0】Django框架之视图层
    【一】三板斧【1】HttpResponseHttpResponse是Django中用于创建HTTP响应对象的类,它允许你构建并返回服务器对客户端请求的数据和状态。当需要直接返回纯文本数据(如JSON格式的数据)或者HTML页面时,可以使用HttpResponse。fromdjango.httpimportHttpResponsedef......
  • 【Django开发】0到1美多商城项目md教程第2篇:展示用户注册页面,1. 创建用户模块子应用
    美多商城完整教程(附代码资料)主要内容讲述:欢迎来到美多商城!,项目准备。展示用户注册页面,创建用户模块子应用。用户注册业务实现,用户注册前端逻辑。图形验证码,图形验证码接口设计和定义。短信验证码,避免频繁发送短信验证码。账号登录,用户名登录。登录,登录开发文档。用户基本信息,查询......
  • Docker部署Django项目——基础
    1.服务器配置1.1centos7系统的安装centos-7-isos-x86_64安装包下载)VMware安装自定义配置选择对应的系统镜像一般选择内核4核、内存8g、硬盘80g相关配置1.2.网络配置1.2.1查看win电脑虚拟机VMnet8的ip使用ipconfig查看虚拟机的ip1.2.2配置虚拟机VMne......