Django ModelForm

Django的ModelForm的验证方式相比较form + Model的验证方式有下列区别:

  • ModelForm没有form + Model的低耦合性
  • ModelForm更适合小型项目,而form + Model则无此限制
  • 都是继承BaseForm类,因此也能使用BaseForm中的三个内置钩子

一、ModelForm简单示例:

1、进行modelForm实例之前先在models中创建两张表

from django.db import models

# Create your models here.

class UserType(models.Model):
caption = models.CharField(max_length=32) class UserInfo(models.Model):
username = models.CharField(max_length=32)
email = models.EmailField()
user_type = models.ForeignKey(to="UserType",to_field='id',on_delete=models.CASCADE)

2、在form.py中创建ModelForm

from app01 import models
from django import forms
from django.forms import fields class UserInfoModelForm(forms.ModelForm): #还是在forms里面,需要继承modelform class Meta:
model = models.UserInfo #表示去哪一个类里面去获取字段
fields = '__all__' #能获取字段,也可以对这张表进行增删改查,all表示展示所有列
#fields = ['username',] #只选择username列,这个是UserInfo中的字段
#exclude = ["username"] #排除username字段,展示其他字段

3、在view.py中设置验证函数

def index(request):
if request.method == "GET":
obj = UserInfoModelForm() #验证时,创建modelform对象
return render(request,'index.html',{'obj':obj})
elif request.method == "POST":
obj = UserInfoModelForm(request.POST) #在views函数里面做校验
return render(request,'index.html',{'obj':obj})

4、templas中的HTML模板设置

<body>
<form>
{{ obj.as_p }}
<input type="submit" value="提交"/>
</form>
</body>

二、ModelForm中的内置组件、数据保存、示例

1、ModelForm除modle、fields、exclude组件外,其他组件介绍:

ModelForm
class Meta:
model, # 对应Model的
fields=None, # 字段
exclude=None, # 排除字段
labels=None, # 提示标签设置
"""
labels = {
'username':'用户名',
'email':'邮箱',
}
"""
help_texts=None, # 输入框后的帮助提示信息
"""
help_texts = {
'username':'....',
'email':'xxx@126.com',
} """
widgets=None, # 自定义插件
"""
from django.forms import widgets as Fwidgets #需要导入html插件,并且需要重新命名,因为跟关键字重复了 #html插件
widgets = {
"username":Fwidgets.Textarea(attrs={'class':'c1'}) #加上属性
}
"""
error_messages=None, # 自定义错误信息(整体错误信息from django.core.exceptions import NON_FIELD_ERRORS)
"""
error_messages = {
"__all__":{
#定义整体的错误信息
},
'email':{
'required':'邮箱不为空', # 根据required是code值
'invalid':"邮箱格式错误"
}
}
"""
field_classes=None # 自定义字段类 (也可以自定义字段)
"""
#把标签的格式转换为其他格式,这边需要导入fields,但是需要重新起一个别名不然会跟自身的fields冲突,就会报错
from django.forms import fields as Ffields #导入fields,并且取一个别名
field_classes = {
'email':Ffields.URLField #把email格式转换为url格式
}
"""
localized_fields=('birth_date',) # 本地化,如:根据不同时区显示数据
"""
如:
数据库中
2016-12-27 04:10:57
setting中的配置
TIME_ZONE = 'Asia/Shanghai'
USE_TZ = True
则显示:
2016-12-27 12:10:57
""" # ##自定义字段
"""
#modelform可以自定义一些字段,比如10天免登录,这个东西不需要提交到后台的
class UserInfoModelForm(forms.ModelForm):
#自定义字段
is_rmb = Ffields.CharField(
widget=Fwidgets.CheckboxInput() #modelform可以自定义一些字段,比如10天免登录,这个东西不需要提交给后台的
) class Meta:
model = models.UserInfo fields = "__all__" """

2、models下ForeignKey、ManyToMany的数据保存

 class UserType(models.Model):
caption = models.CharField(max_length=32) class UserGroup(models.Model):
name = models.CharField(max_length=32) class UserInfo(models.Model):
username = models.CharField(max_length=32)
email = models.EmailField()
user_type = models.ForeignKey(to="UserType",to_field='id',on_delete=models.CASCADE)
u2g = models.ManyToManyField(UserGroup) #跟上面的UserGroup建立多对多关系

models

#1对1的外键,验证通过后,save()一下,数据就直接可以保存在数据库了
def index(request):
if request.method == "GET":
obj = UserInfoModelForm()
return render(request,'index.html',{'obj':obj})
elif request.method == "POST":
obj = UserInfoModelForm(request.POST)
if obj.is_valid():
obj.save() #验证成功之后,是直接保存在数据库中
"""
form + model中的验证函数
# print(obj.is_valid())
# print(obj.cleaned_data)
# print(obj.errors.as_json())
"""
return render(request,'index.html',{'obj':obj})

ForeignKey

 def index(request):
if request.method == "GET":
obj = UserInfoModelForm()
return render(request,'index.html',{'obj':obj})
elif request.method == "POST":
obj = UserInfoModelForm(request.POST)
if obj.is_valid():
# commit为true时将会同时保存回将UserInfo表和其多对多关系表同时进行保存,为false时什么都不会做,需要自己手动保存UserInfo表和其多对多关系表
obj.save(commit=True) instance = obj.save(commit=False) # 什么保存操作都不回做
instance.save() # 这个不等于obj.save(),它只保存当前这个表数据,不会保存其多对多关系表
obj.save_m2m() # 保存其多对多关系表
return render(request,'index.html',{'obj':obj})

ManyToMany

3、示例

 from app01 import models

 def user_list(request):
li = models.UserInfo.objects.all().select_related('user_type') #select_related 只能是外键,不能是多对多
return render(request,'user_list.html',{'li':li}) def user_edit(request,nid):
if request.method == "GET":
user_obj = models.UserInfo.objects.filter(id=nid).first()
mf = UserInfoModelForm(instance=user_obj) #把对象传进来,在前端就能显示默认值
return render(request,'user_edit.html',{'mf':mf,'nid':nid})
elif request.method == "POST":
user_obj = models.UserInfo.objects.filter(id=nid).first()
mf = UserInfoModelForm(request.POST,instance=user_obj) #我把数据提交过来了,并且我拿到对哪个对象进行更新数据
if mf.is_valid():
mf.save()
else:
print(mf.errors.as_json())
return render(request,'user_edit.html',{'mf':mf,'nid':nid})

view.py

<body>
<ul>
{% for row in li %}
<li>{{ row.username }}-{{ row.user_type.caption }}-<a href="/edit-{{ row.id }}/">编辑</a></li>
{% endfor %}
</ul>
</body>

user_list.html

<body>
<form method="post" action="/edit-{{ nid }}/">
{{ mf.as_p }}
<input type="submit" value="提交"/>
</form>
</body>

user_edit.html

【python】-- Django ModelForm的更多相关文章

  1. 【python】-- Django Form

    Django  Form Django的Form主要具有一下几大功能: 生成HTML标签 验证用户数据(显示错误信息) HTML Form提交保留上次提交数据 初始化页面显示内容(自定义样式) 一.F ...

  2. 【python】-- Django ORM(基础)

    Django ORM(基础) Django 框架十分强大,自带数据库操作功能.Django 跟 SQLAchemy 一样,也是通过ORM(Object Relational Mapping,关系对象映 ...

  3. 【python】-- Django

    Django Python的WEB框架有Django.Tornado.Flask 等多种,Django相较与其他WEB框架其优势为:大而全,框架本身集成了ORM.模型绑定.模板引擎.缓存.Sessio ...

  4. 【Python】django安装

    官方下载:https://www.djangoproject.com/download/ 报错 [root@test Django-]# python setup.py install Traceba ...

  5. 【python】-- Django 中间件、缓存、信号

    Django  中间件.缓存.信号 一. Django  中间件 django 中的中间件(middleware),在django中,中间件其实就是一个类,在请求到来和结束后,django会根据自己的 ...

  6. 【python】-- Django ORM(进阶)

    Django ORM(进阶) 上一篇博文简述了Django ORM的单表操作,在本篇博文中主要简述Django ORM的连表操作. 一.一对多:models.ForeignKey() 应用场景:当一张 ...

  7. 【Python】Django

    数据表更改 Django 1.7.x 和后来的版本: Django 1.7.x 及以后的版本集成了 South 的功能,在修改models.py了后运行: python manage.py makem ...

  8. 【Python】Django支持事务方式

    代码: with transaction.atomic(): for i in xrange(int(svc_instance_num)): tmp_fileprotect_svc_instance ...

  9. 【Python】Django 时间字段 最佳实践

    . python datetime from datetime import datetime datetime.now() datetime.utcnow() from datetime impor ...

随机推荐

  1. 【BIEE】17_仪表盘提示中值按顺序显示

    我们在使用仪表盘提示的时候,常常会遇到这种问题,客户需要将一些常见的放在最上边,这样方便显示 例如:现在有数据库表如下 我们需要在提示中展示机构名称,我们如下新建提示 展示效果如下: 我们从上图可以看 ...

  2. Refactoring之——代码的坏味道(一)过长方法

    1 代码的坏味道 重构一书中提到了22种代码的坏味道,大致可以分为几类. 识别代码的坏味道,有助于发现代码的潜在问题,从而可以有的放矢的修改现有代码,使之不断完善. 1.1 Bloaters(臭鲱,暂 ...

  3. node.js零基础详细教程(2):模块化、fs文件操作模块、http创建服务模块

    第二章  建议学习时间4小时  课程共10章 学习方式:详细阅读,并手动实现相关代码 学习目标:此教程将教会大家 安装Node.搭建服务器.express.mysql.mongodb.编写后台业务逻辑 ...

  4. win10下iis绑定局域网ip无效的解决方案

    win7不会出现此问题 win10会 win8未测试 问题描述 <binding protocol="http" bindingInformation="*:808 ...

  5. linux 自启动

    使用chkconfig命令可以查看在不同启动级别下课自动启动的服务(或是程序),命令格式如下:chkconfig --list可能输出如下:openvpn 0:关闭 1:开启 ...... 6:关闭 ...

  6. SGDMA-----Scatter-gather DMA

    Scatter-gather DMA 使用一个链表描述物理上不连续的存储空间,然后把链表首地址告诉DMA master.DMA master在传输完一块物理连续的数据后,不用发起中断,而是根据链表来传 ...

  7. PCIE博文链接

    http://blog.csdn.net/mao0514/article/category/1518607/1

  8. Python内置函数之any()

    any()函数和all()函数相对立. 相同点为: any()也只能传入一个参数. any()的参数必须是可迭代对象. 不同点: 可迭代对象中只要有一个元素为True,返回值就是True. 下面看看具 ...

  9. phpExcel常用方法详解【附有php导出excel加超级链接】

    phpExcel常用方法详解[附有php导出excel加超级链接] 发表于4年前(-- :) 阅读() | 评论() 0人收藏此文章, 我要收藏 赞0 http://www.codeplex.com/ ...

  10. vue实现结账单基本方法

    <script> import axios from 'axios'; export default { name: 'Pos', mounted: function () { var o ...