Django基础核心技术之Model模型的介绍与设计
原创: Yunbo Shi Python Web与Django开发 2018-05-03
Django网络应用开发的5项基础核心技术包括模型(Model)的设计,URL的配置,View(视图)的编写,Template(模板)的设计和Form(表单)的使用。今天小编我就拼了老命来用千字长文给你来介绍下第一项Django核心基础知识之Model的设计吧。想持续了解后续Django Web开发技术请订阅我的公众号【Python与Django大咖之路】。

什么是Model模型?

Model (模型) 简而言之即数据模型。模型不是数据本身(比如数据库里的数据),而是抽象的描述数据的构成和逻辑关系。每个Django model实际上是个类,继承了models.Model。每个Model应该包括属性,关系(比如单对单,单对多和多对多)和方法。当你定义好Model模型后,Django的接口会自动帮你在数据库生成相应的数据表(table)。这样你就不用自己用SQL语言创建表格或在数据库里操作创建表格了,是不是很省心?

我们来看个书与出版社的实际案例。出版社有名字和地址。书有名字,描述和添加日期。当然我们还要利用ForeignKey定义出版社与书单对多的关系,因为一个出版社可以出版很多书。我们定义了如下模型,那你看看代码有问题吗?

models.py
from django.db import models

class Publisher(models.Model):
name = models.CharField(max_length=30)
address = models.CharField()

def __str__(self):
return self.name
1
2
class Book(models.Model):
name = models.CharField(max_length=30)
description = models.TextField(blank=True, null=True)
publisher = ForeignKey(Publisher)
add_date = models.DateField()

def __str__(self):
return self.name
1
2
当你运行python manage.py migrate 创建表格的时候你会遇到错误,错误原因如下:

CharField里的max_length选项没有定义

ForeignKey(Publisher)里的on_delete选项有没有定义

所以当你定义Django模型Model的时候,你一定要十分清楚2件事:

这个Field是否有必选项, 比如CharField的max_length和ForeignKey的on_delete选项是必须要设置的。

这个Field是否必需(blank = True or False),是否可以为空 (null = True or False)。这关系到数据的完整性。

其实在上述案例中还有一个隐藏的错误,即TextField(blank = True, null = True)。blank = True 意味这个字段不是必需的,在客户端不是必填选项。null = True意味这个数据库里这个字段可以存储为null空值。但是Django对于空白的CharField和TextField永远不会存为null空值,而是存储空白字符串’’,所以正确的做法是设置default=’’。

下表才是一个比较正确的Django模型(Model)。

models.py
from django.db import models

class Publisher(models.Model):
name = models.CharField(max_length=30)
address = models.CharField(max_length=60)

def __str__(self):
return self.name
1
2
class Book(models.Model):
name = models.CharField(max_length=30)
description = models.TextField(blank=True, default=’’)
publisher = ForeignKey(Publisher, on_delete = models.CASCADE)
add_date = models.DateField()

def __str__(self):
return self.name
1
2
Django Model中字段(Field)的可选项和必选项

字段与选项(必选项为黄色标注)
CharField() 字符字段

max_length = xxx or None

如不是必填项,可设置blank = True和default = ‘’

如果用于username, 想使其唯一,可以设置unique = True

如果有choice选项,可以设置 choices = XXX_CHOICES

TextField() 文本字段

max_length = xxx

如不是必填项,可设置blank = True和default = ‘’

DateField() and DateTimeField() 日期与时间字段

一般建议设置默认日期default date.

For DateField: default=date.today - 先要from datetime import date

For DateTimeField: default=timezone.now - 先要from django.utils import timezone

对于上一次修改日期(last_modified date),可以设置: auto_now=True

EmailField() 邮件字段

如不是必填项,可设置blank = True和default = ‘’

一般Email用于用户名应该是唯一的,建议设置unique = True

IntegerField(), SlugField(), URLField(),BooleanField()

可以设置blank = True or null = True

对于BooleanField一般建议设置defautl = True or False

FileField(upload_to=None, max_length=100) - 文件字段

upload_to = “/some folder/”

max_length = xxxx

ImageField(upload_to=None, height_field=None, width_field=None, max_length=100,)

upload_to = “/some folder/”

其他选项是可选的.

ForeignKey(to, on_delete, **options) - 单对多关系

to必需指向其他模型,比如 Book or ‘self’ .

必需指定on_delete options(删除选项): i.e, “on_delete = models.CASCADE” or “on_delete = models.SET_NULL” .

可以设置"default = xxx" or “null = True” .

如果有必要,可以设置 "limit_choices_to = ",如下面例子。

staff_member = models.ForeignKey( User, on_delete=models.CASCADE, limit_choices_to={‘is_staff’: True}, )

可以设置 “related_name = xxx” 便于反向查询。

ManyToManyField(to, **options) - 多对多关系

to 必需指向其他模型,比如 User or ‘self’ .

设置 "symmetrical = False " if 多对多关系不是对称的

设置 "through = ‘intermediary model’ " 如果需要建立中间模型来搜集更多信息

可以设置 “related_name = xxx” 便于反向查询。

一个复杂点的Django Model模型

我们现在来看一个更复杂点的Django模型。假设我们要开发一个餐厅(restaurant)的在线点评网站,允许用户(user)上传菜肴(dish)的图片并点评餐厅,我们就可以设计如下模型。用户与餐厅,餐厅与菜肴,及用户与菜肴都是单对多的关系。我们可以这样理解:一个用户可以访问点评多个餐厅,一个餐厅有多个菜肴,一个用户可以上传多个菜肴的图片。

models.py
from django.db import models
from django.contrib.auth.models import User
from datetime import date

class Restaurant(models.Model):
name = models.TextField()
address = models.TextField(blank=True, default=’’)
telephone = models.TextField(blank=True, default=’’)
url = models.URLField(blank=True, null=True)
user = models.ForeignKey(User, default=1,on_delete=models.CASCADE)
date = models.DateField(default=date.today)

def __str__(self):
return self.nameclass Dish(models.Model):
name = models.TextField()
description = models.TextField(blank=True, default='')
price = models.DecimalField('USD amount', max_digits=8, decimal_places=2, blank=True, null=True)
user = models.ForeignKey(User, default=1, on_delete=models.CASCADE)
date = models.DateField(default=date.today)
image = models.ImageField(upload_to="myrestaurants", blank=True, null=True)
1
2
3
4
5
6
7
8
Related name “dishes” allows you to use restaurant.dishes.all to access all dishes objects
instead of using restaurant.dish_set.all
restaurant = models.ForeignKey(Restaurant, null=True, related_name='dishes', on_delete=models.CASCADE)

def __str__(self):
return self.name
1
2
3
4
This Abstract Review can be used to create RestaurantReview and DishReview
class Review(models.Model):
RATING_CHOICES = ((1, ‘one’), (2, ‘two’), (3, ‘three’), (4, ‘four’), (5, ‘five’))
rating = models.PositiveSmallIntegerField(‘Rating’, blank=False, default=3, choices=RATING_CHOICES)
comment = models.TextField(blank=True, null=True)
user = models.ForeignKey(User, default=1, on_delete=models.CASCADE)
date = models.DateField(default=date.today)

class Meta:
abstract = True
1
2
class RestaurantReview(Review):
restaurant = models.ForeignKey(Restaurant, on_delete=models.CASCADE)

def __str__(self):
return "{} review".format(self.restaurant.name)
1
2
你观察到Django模型设计里的细节了吗?

我们的Dish模型里有一个restaurant的字段,建立了一个单对多的关系。我们可以通过dish.restaurant.name直接查询到菜肴所属的餐厅的名字。然而我们Restaurant模型里并没有dish的字段,我们如何根据restaurant查询到某个餐厅的所有菜肴呢?Django非常聪明,可以通过在dish小写后面加上’_set’进行反向查询。我们本来可以直接通过restaurant.dish_set.all的方法来进行查找的,然而这个方法并不直观。为了解决这个问题,我们在dish模型里设置’related_name = dishes", 这样我们就可以直接通过restaurant.dishes.all来反向查询所有菜肴了。注意一但你设置了related name, 你将不能再通过_set方法来反向查询。

restaurant = models.ForeignKey(Restaurant, related_name=‘dishes’, on_delete=models.CASCADE)

第2个细节你需要关注的是Review模型里,我们设置了META选项: Abstract = True。这样一来Django就会认为这个模型是抽象类,而不会在数据库里创建review的数据表。实际上Model自带的META选项还有很多,都非常有用。见下文。

常见的Django Model META类选项

models.py
from django.db import models

class Meta:
# 按Priority降序, order_date升序排列.
get_latest_by = [’-priority’, ‘order_date’]
# 自定义数据库里表格的名字
db_table = ‘music_album’
# 按什么排序
ordering = [‘pub_date’]
# 定义APP的标签
app_label = ‘myapp’
# 声明此类是否为抽象
abstract = True
# 添加授权
permissions = ((“can_deliver_pizzas”, “Can deliver pizzas”),)

转载于:https://www.cnblogs.com/ly570/p/10992439.html

Django基础核心技术之Model模型的介绍与设计相关推荐

  1. python django框架分析_Django框架模型简单介绍与使用分析

    本文实例讲述了Django框架模型简单介绍与使用.分享给大家供大家参考,具体如下: ORM介绍 ORM Object relational mapping 对象关系映射 把面向对象中的类和数据库表一一 ...

  2. Django基础02 视图和模型

    2. Django的V和M 2.1 视图(views) 视图是Django程序中处理后端业务逻辑的地方. Django的视图是定义在子应用的views.py中的. Django的视图分为 函数视图 和 ...

  3. Django基础5.1,模型层(二)多表操作

    Django5.1 创建模型 建立下列模型 # 作者表 class Author(models.Model):name=models.CharField( max_length=32)age=mode ...

  4. NLP基础核心技术(语法模型和语法树)

    1,写入语法规则 simple_grammar = """ sentence => noun_phrase verb_phrase #句子语法结构:名词后面接上动词 ...

  5. Django基础(11): 表单集合Formset的高级用法详解

    Formset(表单集)是多个表单的集合.Formset在Web开发中应用很普遍,它可以让用户在同一个页面上提交多张表单,一键添加多个数据,比如一个页面上添加多个用户信息.今天小编我就介绍下Djang ...

  6. Django基础(6): 模型Models高级进阶必读。

    你或许已经早已读过我们的原创文章Django基础(1): 模型Models的介绍与设计,并已经知道一个模型的设计是一个app的核心.然而仅知道基础知识是远远不够的,在实际web开发过程中你需要掌握一些 ...

  7. 语音识别学习日志 2019-7-14 语音识别基础知识准备3 {Kmean算法分析与HMM(Hidden Markov Model)模型}

    Kmean算法 聚类算法 对于"监督学习"(supervised learning),其训练样本是带有标记信息的,并且监督学习的目的是:对带有标记的数据集进行模型学习,从而便于对新 ...

  8. Django基础五之Django模型层(二)多表操作

    Django基础五之Django模型层(二)多表操作 一 创建模型 表和表之间的关系 一对一.多对一.多对多 # 作者表 比较常用的信息放在这个表中 class Author(models.Model ...

  9. Python经典前端框架:Django,第三天【Django基础教学--模型】

    Django基础教学--模型 Django基础教学--模型 操作环境 安装第三方库 前言 模型定义: 连接Mysql数据库的一些配置 连接数据库重点 创建模型 记录模型操作 为什么有这个文件 同步到数 ...

最新文章

  1. 神经网络变成小怪兽,还能互相 battle!这款游戏用最简单的方式训练神经网络...
  2. 使用最新目标跟踪框mmtracking实现自己的目标跟踪项目
  3. NGcodec谈FPGA编码与HEVC和AV1
  4. Linux下修改swap的大小
  5. Linux 死机了怎么办
  6. 去除html重复的元素 js,js数组中去除重复值的几种方法
  7. java安装_JAVA 安装
  8. 悲哀!面试现场,简单几道java算法题,90%程序员没写出来
  9. Python list 初始化技巧
  10. [转载] kotlin 字符串_Kotlin基本类型字符串
  11. python ndarray转binary_Python 实现Image和Ndarray互相转换
  12. java session 使用_浅谈Session的使用(原创)
  13. java转账_JAVA实现银行转账功能
  14. Styler类的变量
  15. linux 变量替换字符串,shell中常用的变量处理、字符串操作(之一)
  16. 蓝桥杯真题系列:第十一届蓝桥杯C语言B组集锦
  17. Maven环境配置(本文以“apache-maven-3.8.2”的安装配置为例)
  18. 计算机专业助我成长作文600,电脑伴我成长作文
  19. fir.im Weekly - 当技术成为一种“武器”
  20. 查找 android 恶意软件,安卓恶意软件分析: 剖析 Hydra Dropper

热门文章

  1. 4.Vue 模板语法
  2. 【Tools】虚拟串口工具之VSPD10
  3. 【STM32】FreeRTOS任务相关API
  4. serverless 框架_Malaguv1.4.1支持Serverless的微服务框架
  5. Python入门--模块的导入和使用
  6. 每天一道LeetCode-----以字符串的形式输出二叉树所有从根节点到叶子节点的路径
  7. C++学习笔记-----函数调用时的决议:名字查找,重载决议,可访问性检测
  8. 翻转字符串里的单词—leetcode151
  9. 环形链表—leetcode141
  10. iOS 文件预览 UIDocumentInteractionController