首先上结论:

  form适用于对单个表单的操作,并且需要对每个字段的验证规则自定义。

  modelform:适用于对用户提交的单个表单操作,字段可以用model中的表的字段来作为验证规则,适用于快速的进行增加、修改。

  formset:适用于对多个表单进行操作,字段需要也可以用model中的表的字段来作为验证规则。

  modelfoemset:适用于对多个表单进行操作,字段需要也可以用model中的表的字段来作为验证规则,速度可能快一些(有待考证)

form用法

后端代码

from django.forms import widgets

wid_01=widgets.TextInput(attrs={"class":"form-control"})
wid_02=widgets.PasswordInput(attrs={"class":"form-control"}) class UserForm(forms.Form):
name=forms.CharField(max_length=32,
widget=wid_01
)
pwd=forms.CharField(max_length=32,widget=wid_02)
r_pwd=forms.CharField(max_length=32,widget=wid_02)
email=forms.EmailField(widget=wid_01)
tel=forms.CharField(max_length=32,widget=wid_01) def register(request): if request.method=="POST":
form=UserForm(request.POST)
if form.is_valid():
print(form.cleaned_data) # 所有干净的字段以及对应的值
else:
print(form.cleaned_data) #
print(form.errors) # ErrorDict : {"校验错误的字段":["错误信息",]}
print(form.errors.get("name")) # ErrorList ["错误信息",]
return HttpResponse("OK")
form=UserForm()
return render(request,"register.html",locals())
<form action="" method="post">
{% csrf_token %} {% for field in form %}
<div>
<label for="">{{ field.label }}</label>
{{ field }}
</div>
{% endfor %}
<input type="submit" class="btn btn-default pull-right"> </form>

modelform用法

class MenuModelForm(forms.ModelForm):
class Meta:
model = models.Menu
fields = '__all__'
widgets = {
'title': forms.TextInput(attrs={'placeholder': '请输入角色名称', 'class': 'form-control'}),
'icon': forms.RadioSelect(
choices=ICON_LIST
)
}
error_messages = {
'title': {
'required': '菜单名称不能为空',
},
'icon': {
'required': '请选择图标',
}
} def menu_add(request):
"""
添加菜单
:param request:
:return:
"""
if request.method == 'GET':
form = MenuModelForm()
else:
form = MenuModelForm(request.POST)
if form.is_valid():
print(form.data)
form.save()
return redirect(reverse('rbac:menu_list'))
return render(request, 'rbac/menu_change.html', {'form': form}) def menu_edit(request, pk):
"""
编辑菜单
:param request:
:return:
"""
obj = models.Menu.objects.filter(id=pk).first()
if not obj:
return HttpResponse('菜单不存在') if request.method == 'GET':
form = MenuModelForm(instance=obj)
return render(request, 'rbac/menu_change.html', {'form': form}) form = MenuModelForm(request.POST, instance=obj)
if form.is_valid():
form.save()
return redirect(reverse('rbac:menu_list'))
<div class="luffy-container">
<form class="form-horizontal" method="post" novalidate>
{% csrf_token %}
<div class="form-group">
<label class="col-sm-2 control-label">菜单名称:</label>
<div class="col-sm-6">
{{ form.title }} </div>
<div class="col-sm-offset-1 col-sm-2">
<input type="submit" value="保 存" class="btn btn-primary">
</div>
</div>
<div class="form-group">
<label class="col-sm-2 control-label">图标:</label>
<div class="col-sm-7 icon-area">
{{ form.icon }}
</div>
<div class="col-sm-3">
<div>{{ form.title.errors.0 }}</div>
<div>{{ form.icon.errors.0 }}</div>
</div>
</div> </form>
</div>

formset用法

class MultiPermissionForm(forms.Form):
id = forms.IntegerField(
widget=forms.HiddenInput(),
required=False
)
title = forms.CharField(
widget=forms.TextInput(attrs={'class': "form-control"})
)
url = forms.CharField(
widget=forms.TextInput(attrs={'class': "form-control"})
)
name = forms.CharField(
widget=forms.TextInput(attrs={'class': "form-control"})
)
menu_id = forms.ChoiceField(
choices=[(None, '-----')],
widget=forms.Select(attrs={'class': "form-control"}),
required=False, ) pid_id = forms.ChoiceField(
choices=[(None, '-----')],
widget=forms.Select(attrs={'class': "form-control"}),
required=False,
) def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
self.fields['menu_id'].choices += models.Menu.objects.values_list('id', 'title')
self.fields['pid_id'].choices += models.Permission.objects.filter(pid__isnull=True).exclude(
menu__isnull=True).values_list('id', 'title') def clean_pid_id(self):
menu = self.cleaned_data.get('menu_id')
pid = self.cleaned_data.get('pid_id')
if menu and pid:
raise forms.ValidationError('菜单和根权限同时只能选择一个')
return pid def add(request):
"""
增加
:param request:
:return:
"""
MultiPermissionFormSet = formset_factory(MultiPermissionForm, extra=0)
if request.method == 'GET':
form = MultiPermissionFormSet()
return render(request,'list.html',{'form':form}
form = MultiPermissionFormSet(request.post)
if form.is_valid():
return redict('url')
else:
......... def edit(request):
"""
修改
:param request:
:return:
"""
MultiPermissionFormSet = formset_factory(MultiPermissionForm, extra=0)
if request.method == 'GET':
form = MultiPermissionFormSet('数据')
# formset是支持批量修改的所以 这里传入的数据一定要是可迭代对象,并且里面的
数据类型是字典或者对象
return render(request,'list.html',{'form':form}
form = MultiPermissionFormSet(request.post)
if form.is_valid():
return redict('url')
else:
.........
<div class="luffy-container">
<form method="post" action="?type=generate">
{% csrf_token %}
{{ form.management_form }}
# 这里一定要记得写这一步
<div class="panel panel-default">
<!-- Default panel contents -->
<div class="panel-heading">
<i class="fa fa-binoculars" aria-hidden="true"></i> 待新建权限列表
<button class="right btn btn-primary btn-xs" style="padding: 2px 8px;margin: -3px;">
<i class="fa fa-save" aria-hidden="true"></i>
新建
</button>
</div>
<div class="panel-body" style="color: #9d9d9d;">
注意:路由系统中自动发现且数据库中不存在的路由。
</div> <table class="table table-bordered">
<thead>
<tr>
<th>序号</th>
<th>名称</th>
<th>URL</th>
<th>别名</th>
<th>所属菜单</th>
<th>根权限</th>
</tr>
</thead>
<tbody>
{% for form in generate_formset %} <tr>
<td style="vertical-align: middle;">{{ forloop.counter }}</td>
{% for field in form %} # 这里在渲染标签的时候一定要渲染 id这个标签 这个很重要不然会报错 这里是写了两个fou循环 所以自动渲染了id标签
{% if forloop.first %}
<td class="hide">
{% else %}
<td>
{% endif %}
{{ field }}<span style="color: firebrick;">{{ field.errors.0 }}</span>
</td>
{% endfor %}
{% endfor %}
</tbody>
</table>
</div>
</form>

modelformset用法

class StudyRecordModelForm(forms.ModelForm):
class Meta:
model = models.StudyRecord
fields = ['student','record','score','homework_note'] def changelist_view(self,request):
ccid = request.GET.get('ccid')
model_formset_cls = modelformset_factory(models.StudyRecord,StudyRecordModelForm,extra=0)
queryset = models.StudyRecord.objects.filter(course_record_id=ccid)
if request.method == "GET":
formset = model_formset_cls(queryset=queryset)
        # 这里UI定是个可迭代对象,因为modelformset是操作多表的,里面的数据类型可以为字典或者对象
return render(request,'study_record.html',{'formset':formset}) formset = model_formset_cls(data=request.POST)
print(request.POST)
if formset.is_valid():
formset.save()
return redirect('/stark/crm/studyrecord/list/?ccid=%s' %ccid )
return render(request, 'study_record.html', {'formset': formset})
  <div class="panel panel-default">
<div class="panel-heading">学习记录</div>
<div class="panel-body">
<div style="width: 680px;margin: 0 auto;">
<form method="post">
{% csrf_token %}
{{ formset.management_form }}
# 这里一定要加这句代码 <table class="table table-bordered">
<thead>
<tr>
<th>姓名</th>
<th>考勤</th>
<th>作业成绩</th>
<th>作业评语</th>
</tr>
</thead>
<tbody>
{% for form in formset %}
<tr>
{{ form.id }}
# 这里只写了一层for循环,所以手动写字段,必须把id字段写上
<td>{{ form.instance.student }}</td>
<td>{{ form.record }} {{ form.record.errors.0 }}</td>
<td>{{ form.score }} {{ form.score.errors.0 }}</td>
<td>{{ form.homework_note }} {{ form.homework_note.errors.0 }}</td>
</tr>
{% endfor %}
</tbody>
</table>
<input type="submit" value="保存">
</form>
</div>
</div>
</div>

以上为这四种的区别和用法。

Django之路12——form modelform formset modelformset的各种用法的更多相关文章

  1. form modelform formset modelformset的各种用法

    form modelform formset modelformset的各种用法   首先上结论: form适用于对单个表单的操作,并且需要对每个字段的验证规则自定义. modelform:适用于对用 ...

  2. django基础 -- 10.form , ModelForm ,modelformset

    一.生成页面可用的 HTML标签 1.form 所有内置字段 Field required=True, 是否允许为空 widget=None, HTML插件 label=None, 用于生成Label ...

  3. 12.Django基础十之Form和ModelForm组件

    一 Form介绍 我们之前在HTML页面中利用form表单向后端提交数据时,都会写一些获取用户输入的标签并且用form标签把它们包起来. 与此同时我们在好多场景下都需要对用户的输入做校验,比如校验用户 ...

  4. Django基础十之Form和ModelForm组件

    一 Form介绍 我们之前在HTML页面中利用form表单向后端提交数据时,都会写一些获取用户输入的标签并且用form标签把它们包起来. 与此同时我们在好多场景下都需要对用户的输入做校验,比如校验用户 ...

  5. {Django基础十之Form和ModelForm组件}一 Form介绍 二 Form常用字段和插件 三 From所有内置字段 四 字段校验 五 Hook钩子方法 六 进阶补充 七 ModelForm

    Django基础十之Form和ModelForm组件 本节目录 一 Form介绍 二 Form常用字段和插件 三 From所有内置字段 四 字段校验 五 Hook钩子方法 六 进阶补充 七 Model ...

  6. day 64 Django基础十之Form和ModelForm组件

    Django基础十之Form和ModelForm组件   本节目录 一 Form介绍 二 Form常用字段和插件 三 From所有内置字段 四 字段校验 五 Hook钩子方法 六 进阶补充 七 Mod ...

  7. Django之Model、Form、ModelForm区别

    本节内容: 1:Model               https://www.cnblogs.com/shuai1991/p/10844662.html 2:Form                 ...

  8. python 学习笔记十七 django深入学习二 form,models

    表单 GET 和 POST 处理表单时候只会用到GET 和 POST 方法. Django 的登录表单使用POST 方法,在这个方法中浏览器组合表单数据.对它们进行编码以用于传输.将它们发送到服务器然 ...

  9. Django之路: 基本命令与网址进阶

    一.Django 基本命令 温馨提示:如果你想学习Django,那么就请您从现在开始按照笔记记录一步一步的用手把代码敲出来,千万不要偷懒哦..... 1.创建一个Django project djan ...

随机推荐

  1. RNN(3) ------ “blog:RNN学习之路”

    博客链接:http://blog.csdn.net/yangyangyang20092010/article/details/50374289 Recurrent Neural Network 学习之 ...

  2. MR目录结构

    D:\MRData\MR\MRS\2017-05-25\TD-LTE_MRS_ZTE_OMC1_20170425000000.zip 每个zip中包含若干FDD-LTE_MRS_OMC1_28163_ ...

  3. kafka系列六、java管理kafka Topic

    package com.example.demo.topic; import kafka.admin.AdminUtils; import kafka.admin.RackAwareMode; imp ...

  4. 调用链系列三、基于zipkin调用链封装starter实现springmvc、dubbo、restTemplate等实现全链路跟踪

    一.实现思路 1.过滤器实现思路 所有调用链数据都通过过滤器实现埋点并收集.同一条链共享一个traceId.每个节点有唯一的spanId. 2.共享传递方式 1.rpc调用:通过隐式传参.dubbo有 ...

  5. Fiddler抓包10-会话框添加查看get与post请求类型

    前言 在使用fiddler抓包的时候,查看请求类型get和post每次只有点开该请求,在Inspectors才能查看get和post请求,不太方便.于是可以在会话框直接添加请求方式. 一.添加会话框菜 ...

  6. js字符串转换成数字与数字转换成字符串的实现方法

    转载:点击查看地址 js字符串转换成数字 将字符串转换成数字,得用到parseInt函数.parseInt(string) : 函数从string的开始解析,返回一个整数. 举例:parseInt(' ...

  7. selenium+python谷歌驱动配置

    1.打开chrome 输入 “chrome://version/”来查看chrome版本 2.访问此网站  http://chromedriver.storage.googleapis.com/ind ...

  8. Spring 核心API

    BeanFactory: 这是一个工厂,用于生产任意Bean,采用延迟加载,第一次getBean时才会加载 ApplicationContext: 是BeanFactory的一个子接口,功能更强大(国 ...

  9. Python实现进度条功能

    Python实现进度条功能 import sys, time def progress(percent, width=50): # 设置进度条的宽度 if percent >= 100: # 当 ...

  10. JavaScript实现抽象类与虚方法(六)

    一:什么是js抽象类与虚方法 虚函数是类成员中的概念,是只做了一个声明而未实现的方法,具有虚函数的类就称之为抽象类,这些虚函数在派生类中才被实现.抽象类是不能实例化的,因为其中的虚函数并不是一个完整的 ...