如何在 django 中处理不同的数据类型?
前言
django 通过继承models
类来完成数据库表的创建,其中牵涉了很多数据类型和相关的知识点,这里把最近碰到的记录一下。
Django 中数据类型的处理
首先来看一下最近一个练手项目(用户管理系统)中基础的部门表和用户表的定义。
1 | from django.db import models |
针对上面出现的类型,这里做一些相应地解释。
CharField
CharField
是基础的字符串类型,上述案例中,verbose_name=
参数在许多函数中都出现了,实际上,这个参数主要是便于开发者自己了解每一个字段代表什么含义,由于我们是 Django 开发,因此我们使用注释的方案也可以,具体看个人喜好。对于 CharField 来说,max_length=
是必要的,因为在创建 MySQL 表时,我们需要指定每一个varchar
的字节数。
DecimalField
是 Python 中十进制浮点数的实例。上述例子中含有两个参数,max_digits=
表示数字位数,这是同时包含整数部分和小数部分的位数。decimal_places=
表示小数位数。default=
用于在使用UserInfo.objects.create()
时没有指定account
参数值的时候的默认值。
DateTimeField
DateTimeField
是专门存储时间的字段类型,格式为Y-m-d H:m:s
。当我们要输出这个值时,我们通常要先对其进行格式化。我们使用strftime({format})
来达到这个目的。format 的格式包含以下多种选择。
1 | %y 两位数的年份表示(00-99) |
当然,我们最常用的还是
1 | time.strftime("%Y-%m-%d %H:%m:%s") # 精确到秒的时间记录 |
在 Django 模板语法中,一般是不允许出现括号的。在前端页面,我们需要改写成
1 | <span>{{ time|date:"Y-m-d" }}</span> |
在 Pycharm 中按住Ctrl查看上述案例中date
的源码,可以发现其也是一个函数,只不过使用了特殊的|
符号表示而已。
ForeignKey
ForeignKey
表示的是外键,也就是将表与表之间建立联系的一种方式。在上述案例中,由于我们需要知晓员工的所属部门,因此我们使用外键将depart
字段与Department
中的id
字段相关联。在 Django 自动生成表的时候,depart
字段会被命名成depart_id
。使用ForeignKey
的时候,我们需要指定参数to=
和to_field=
,前者指定与哪张表关联,后者指定关联的字段名。当然了,由于关联的原因,我们还需要指定on_delete=
,这个参数指定当Department
表中的数据删除时,UserInfo
中对应这些数据的记录将要如何操作。models.CASCADE
表示级联删除,即删除所有与被删除数据相关联的记录;还可以选择models.SET_NULL
,这样所有关联数据的该字段将会被置空。但这有一个前提,你需要在参数列表指定该字段可以为空,即
1 | department = models.ForeignKey(to="Department", to_field="id", null=True, blank=True, on_delete=models.SET_NULL) |
在读取数据中,我们需要先获取UserInfo
的记录,再通过外链访问Department
,即
1 | obj = models.UserInfo.objects.fileter({筛选条件}).first() # 获取满足筛选条件的第一个对象 |
显然,这句语句太长了,Django 提供了另一种通过外键查询另一张表的方式,在上述例子中
1 | obj = models.UserInfo.objects.fileter({筛选条件}).first() # 获取满足筛选条件的第一个对象 |
在这种书写方式下,obj.depart
直接返回根据depart_id
查询到的第一个object
。由于我们这个案例中 ID 值唯一,于是直接访问其.title
即可。
于是在前端页面,我们可以通过如下的模板语法访问每一个人对应的部门的名称:
1 | <tbody> |
SmallIntegerField
该数据类型常应用于固定的枚举型,例如上例中的性别。在绝大多数情况下,性别是不会继续添加的,因此我们可以用元组的嵌套来指定这种对应关系:
1 | gender_choices=( |
相比于存储汉字,在数据库中存储短整型占用更少的存储空间。这种情况下,你需要为SmallIntegerField
指定参数choices=
。这个参数接受一个嵌套元组用来指定对应关系。
当然,这会带来一个问题,我们访问obj.gender
时,会输出整型而非我们想要的汉字“男”或“女”,同样的,Django 提供了一个函数:
1 | obj.get_gender_display() # 这个函数名称会随着你对字段名命名的不同而变化 |
在前端页面,你可以使用模板语法来完成这个功能:
1 | {{ obj.get_gender_display }} |
需要注意的是:你依然不需要在函数末尾添加括号。
Django 的 html 模板
这部分不是本文的重点,但也是Django开发中十分重要的一部分,其可以大大增加代码复用率。因此也一并写在这里。
很多时候,许多 HTML 文件都具备一部分的相同结构,例如,所有页面都有页面顶部的导航栏(navigation bar),都需要引入相同的 css 和 js 等文件。Django 提供模板页面来解决这个问题。我们只需要在模板文件中需要被插入的结构加入:
1 |
|
然后再需要使用的页面引入该模板即可,如果模板名称为layout.html
,那么就可以写成这样:
1 | {% extends 'layout.html' %} {% block {模块名称} %} |
渲染该页面时,Django 会自动将其与模板layout.html
拼接。一个模板中可以有多个这样的结构,一个 HTML 文件中也可以由多个这样的结构组成。
后记
这篇文章字不多,但干货占比也挺高的。赶紧写出来也是方便我后续复习和查询使用。~~所以就浅浅地日更了一下。~~那这篇文章就到此结束吧!该休息咯。