Python探索之ModelForm代码详解


Posted in Python onOctober 26, 2017

这是一个神奇的组件,通过名字我们可以看出来,这个组件的功能就是把model和form组合起来,对,你没猜错,相信自己的英语水平。

先来一个简单的例子来看一下这个东西怎么用:

比如我们的数据库中有这样一张学生表,字段有姓名,年龄,爱好,邮箱,电话,住址,注册时间等等一大堆信息,现在让你写一个创建学生的页面,你的后台应该怎么写呢?

首先我们会在前端一个一个罗列出这些字段,让用户去填写,然后我们从后天一个一个接收用户的输入,创建一个新的学生对象,保存

其实,重点不是这些,而是合法性验证,我们需要在前端判断用户输入是否合法,比如姓名必须在多少字符以内,电话号码必须是多少位的数字,邮箱必须是邮箱的格式这些

当然可以一点一点手动写限制,各种判断,这毫无问题,除了麻烦

我们现在有个更优雅(以后在Python相关的内容里,要多用“优雅”这个词,并且养成习惯)的方法:ModelForm

先来简单的,生硬的把它用上,再来加验证条件:

首先导入ModelForm

from django.forms import ModelForm

在视图函数中,定义一个类,比如就叫StudentList,这个类要继承ModelForm,在这个类中再写一个原类Meta(规定写法,并注意首字母是大写的)

在这个原类中,有以下属性(部分):

class StudentList(ModelForm):
  class Meta:
    model = models.Student #对应的Model中的类
    fields = "__all__"   #字段,如果是__all__,就是表示列出所有的字段
    exclude = None     #排除的字段
    labels = None      #提示信息
    help_texts = None    #帮助提示信息
    widgets = None     #自定义插件
    error_messages = None  #自定义错误信息
#error_messages用法:
    error_messages = {
      'name':{'required':"用户名不能为空",},
      'age':{'required':"年龄不能为空",},
    }
#widgets用法,比如把输入用户名的input框给为Textarea
#首先得导入模块
    from django.forms import widgets as wid #因为重名,所以起个别名
    widgets = {
      "name":wid.Textarea(attrs={"class":"c1"}) #还可以自定义属性
    }
#labels,自定义在前端显示的名字

labels= {
      "name":"用户名"
    }

然后在url对应的视图函数中实例化这个类,把这个对象传给前端

def student(request):

  if request.method == 'GET':
    student_list = StudentList()
    return render(request,'student.html',{'student_list':student_list})

然后前端只需要     {{ student_list.as_p }}   一下,所有的字段就都出来了,可以用as_p显示全部,也可以通过for循环这student_list,拿到的是一个个input框,现在我们就不用as_p,手动把这些input框搞出来,as_p拿到的页面太丑。

首先 for循环这个student_list,拿到student对象,直接在前端打印这个student,是个input框
student.label  ,拿到数据库中每个字段的verbose_name ,如果没有设置这个属性,拿到的默认就是字段名

还可以通过student.errors.0 拿到错误信息

有了这些,我们就可以通过bootstrap,自己拼出来想要的样式了

比如:

<body>
  <div class="container" >
    <h1>student</h1>
    <form method="POST" novalidate>
      {% csrf_token %}
{#      {{ student_list.as_p }}#}
      {% for student in student_list %}
        <div class="form-group col-md-6">
         {# 拿到数据字段的verbose_name,没有就默认显示字段名 #}
          <label class="col-md-3 control-label">{{ student.label }}</label>
          <div class="col-md-9" style="position: relative;">{{ student }}</div>
        </div>
      {% endfor %}
      <div class="col-md-2 col-md-offset-10">
        <input type="submit" value="提交" class="btn-primary">
      </div>
    </form>
  </div>
</body>

现在还缺一个input框的form-contral样式,可以考虑在后台的widget里面添加

比如这样:

from django.forms import widgets as wid #因为重名,所以起个别名
    widgets = {
      "name":wid.TextInput(attrs={'class':'form-control'}),
      "age":wid.NumberInput(attrs={'class':'form-control'}),
      "email":wid.EmailInput(attrs={'class':'form-control'})
    }

当然也可以在js中,找到所有的input框,加上这个样式,也行

保存数据的时候,不用挨个取数据了,只需要save一下

def student(request):

  if request.method == 'GET':
    student_list = StudentList()
    return render(request,'student.html',{'student_list':student_list})
  else:
    student_list = StudentList(request.POST)
    if student_list.is_valid():
      student_list.save()
    return redirect(request,'student_list.html',{'student_list':student_list})

编辑数据:

如果不用ModelForm,编辑的时候得显示之前的数据吧,还得挨个取一遍值,如果ModelForm,只需要加一个instance=obj(obj是要修改的数据库的一条数据的对象)就可以得到同样的效果

保存的时候要注意,一定要注意有这个对象(instance=obj),否则不知道更新哪一个数据

代码示例:

from django.shortcuts import render,HttpResponse,redirect
from django.forms import ModelForm
# Create your views here.
from app01 import models
def test(request):
  # model_form = models.Student
  model_form = models.Student.objects.all()
  return render(request,'test.html',{'model_form':model_form})

class StudentList(ModelForm):
  class Meta:
    model = models.Student #对应的Model中的类
    fields = "__all__"   #字段,如果是__all__,就是表示列出所有的字段
    exclude = None     #排除的字段
    labels = None      #提示信息
    help_texts = None    #帮助提示信息
    widgets = None     #自定义插件
    error_messages = None  #自定义错误信息
#error_messages用法:
    error_messages = {
      'name':{'required':"用户名不能为空",},
      'age':{'required':"年龄不能为空",},
    }
#widgets用法,比如把输入用户名的input框给为Textarea
#首先得导入模块
    from django.forms import widgets as wid #因为重名,所以起个别名
    widgets = {
      "name":wid.Textarea
    }
#labels,自定义在前端显示的名字
    labels= {
      "name":"用户名"
    }
def student(request):
  if request.method == 'GET':
    student_list = StudentList()
    return render(request,'student.html',{'student_list':student_list})
  else:
    student_list = StudentList(request.POST)
    if student_list.is_valid():
      student_list.save()
    return render(request,'student.html',{'student_list':student_list})
def student_edit(request,pk):
  obj = models.Student.objects.filter(pk=pk).first()
  if not obj:
    return redirect('test')
  if request.method == "GET":
    student_list = StudentList(instance=obj)
    return render(request,'student_edit.html',{'student_list':student_list})
  else:
    student_list = StudentList(request.POST,instance=obj)
    if student_list.is_valid():
      student_list.save()
    return render(request,'student_edit.html',{'student_list':student_list})

总结

以上就是本文关于Python探索之ModelForm代码详解的全部内容,希望对大家有所帮助。感兴趣的朋友可以继续参阅本站:python编程羊车门问题代码示例、python中requests使用代理proxies方法介绍等,如有不足之处,欢迎留言指出。感谢朋友们对本站的支持!

Python 相关文章推荐
详解Python中的__init__和__new__
Mar 12 Python
Python中用于计算对数的log()方法
May 15 Python
详解Python list 与 NumPy.ndarry 切片之间的对比
Jul 24 Python
Python自定义函数实现求两个数最大公约数、最小公倍数示例
May 21 Python
Pandas_cum累积计算和rolling滚动计算的用法详解
Jul 04 Python
python对绑定事件的鼠标、按键的判断实例
Jul 17 Python
Python PyPDF2模块安装使用解析
Jan 19 Python
python绘制动态曲线教程
Feb 24 Python
Python使用socketServer包搭建简易服务器过程详解
Jun 12 Python
详解Selenium-webdriver绕开反爬虫机制的4种方法
Oct 28 Python
python语言time库和datetime库基本使用详解
Dec 25 Python
Pandas加速代码之避免使用for循环
May 30 Python
启动targetcli时遇到错误解决办法
Oct 26 #Python
Mac中Python 3环境下安装scrapy的方法教程
Oct 26 #Python
python实现分页效果
Oct 25 #Python
python+pyqt实现12306图片验证效果
Oct 25 #Python
python编程羊车门问题代码示例
Oct 25 #Python
python中requests使用代理proxies方法介绍
Oct 25 #Python
python中requests爬去网页内容出现乱码问题解决方法介绍
Oct 25 #Python
You might like
增加反向链接的101个方法 站长推荐
2007/01/31 PHP
php mssql 时间格式问题
2009/01/13 PHP
php将access数据库转换到mysql数据库的方法
2014/12/24 PHP
php发送http请求的常用方法分析
2016/11/08 PHP
JS 文字符串转换unicode编码函数
2009/05/30 Javascript
读jQuery之六 缓存数据功能介绍
2011/06/21 Javascript
解析JavaScript中点号“.”的多义性
2013/12/02 Javascript
JS中怎样判断undefined(比较不错的方法)
2014/03/27 Javascript
js模拟淘宝网的多级选择菜单实现方法
2015/08/18 Javascript
javascript函数自动执行常用方法汇总
2016/03/28 Javascript
JS学习之表格的排序简单实例
2016/05/16 Javascript
Bootstrap前端开发案例一
2016/06/17 Javascript
AngularJS基础 ng-keypress 指令简单示例
2016/08/02 Javascript
JS多文件上传的实例代码
2017/01/11 Javascript
tracking.js页面人脸识别插件使用方法
2020/04/16 Javascript
JS开发自己的类库实例分析
2019/08/28 Javascript
vue 实现 rem 布局或vw 布局的方法
2019/11/13 Javascript
openlayers实现图标拖动获取坐标
2020/09/25 Javascript
[01:02:48]2018DOTA2亚洲邀请赛 4.1 小组赛 A组 LGD vs OG
2018/04/02 DOTA
Python 内置函数complex详解
2016/10/23 Python
快速入门python学习笔记
2017/12/06 Python
python实现百万答题自动百度搜索答案
2018/01/16 Python
python代码 输入数字使其反向输出的方法
2018/12/22 Python
python pytest进阶之xunit fixture详解
2019/06/27 Python
windows中安装Python3.8.0的实现方法
2019/11/19 Python
浅谈Python中os模块及shutil模块的常规操作
2020/04/03 Python
来自全球大都市的高级街头服饰:Pegador
2018/01/03 全球购物
蹦床仓库:Trampoline Warehouse
2018/12/06 全球购物
中专毕业生自荐信范文
2013/11/28 职场文书
会议邀请函范文
2014/01/09 职场文书
有关打架的检讨书
2014/01/25 职场文书
电子装配专业毕业生求职信
2014/04/23 职场文书
俞敏洪北大演讲稿
2014/05/22 职场文书
邓小平文选读书笔记
2015/06/29 职场文书
MATLAB 如何求取离散点的曲率最大值
2021/04/16 Python
详解Go语言Slice作为函数参数的使用
2021/07/02 Golang