Django ForeignKey与数据库的FOREIGN KEY约束详解


Posted in Python onMay 20, 2020

数据库在高并发的场景下使用外键约束会有锁问题并且使用外键会增加运维成本,所以很多公司都规定生产环境的数据库禁止使用外键。

那么不使用外键约束的情况下使用 Django ORM 如何实现关联查询两个表呢?这曾是困扰我很久的一个问题,今天终于找到了答案,写出来分享一下。

Django 的 ForeignKey 和数据库的 FOREIGN KEY 并不一样。Django 的 ForeignKey 是一种逻辑上的两个表的关联关系,可以指定是否使用数据库的 FOREIGN KEY 约束。

在开头提到的场景下,我们可以这样创建两个表对应的 Model,以省和市的关联举例:

# demo/models.py
from django.db import models

class Province(models.Model):
  name = models.CharField(max_length=16)

  def __unicode__(self):
    return self.name

class City(models.Model):
  name = models.CharField(max_length=16)
  province = models.ForeignKey(Province, null=True, on_delete=models.SET_NULL,
                 related_name='cities', db_constraint=False)
  def __unicode__(self):
    return self.name

以上的 models.py 在执行 migrate 时生成的 SQL 如下(MySQL数据库):

CREATE TABLE `demo_city` (`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY, `name` varchar(16) NOT NULL);
CREATE TABLE `demo_province` (`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY, `name` varchar(16) NOT NULL);
ALTER TABLE `demo_city` ADD COLUMN `province_id` integer NULL;
CREATE INDEX `demo_city_province_id_50fffd49` ON `demo_city` (`province_id`);

如果 ForeignKey 不添加db_constraint=False 参数,会在数据库中使用外键约束,生成以下SQL:

ALTER TABLE `demo_city` ADD CONSTRAINT `demo_city_province_id_aff53934_fk_key_province_id` FOREIGN KEY (`province_id`) REFERENCES `demo_province` (`id`);

另外,ForeignKey 的 on_delete 参数默认为 on_delete=models.CASCADE,表示使用数据库的级联删除,使用 on_delete=models.SET_NULL 可以使删除 Province 时将关联的 City 表对应的 province_id 值设为 NULL

使用这种方式不会破坏 Django 的反向关联查询,以下查询仍然会返回正确的结果:

Province.objects.filter(cities__name='xxx')

实际执行的 SQL 为一个 Inner Join 查询:

SELECT `demo_province`.`id`, `demo_province`.`name` FROM `demo_province` INNER JOIN `demo_city` ON (`demo_province`.`id` = `demo_city`.`province_id`) WHERE `demo_city`.`name` = xxx;

补充知识:关于Django模型中中定义auto_now=True 数据库中的时间并没有自动更新

django的orm关于更新数据库的方法有update和save两种方法。

前提在模型中设置了auto_now=True时间戳属性,为了方便数据库自动更新时间,而

使用update更新的记录,数据库中并没有自动更新,达到我的需求。

auto_now=True自动更新,有一个条件,就是要通过django的model层。

如create或是save方法。

如果是filter之后update方法,则直接调用的是sql,不会通过model层,

所以不会自动更新此时间。所以使用save方法更新才能达到我的需求。

以上这篇Django ForeignKey与数据库的FOREIGN KEY约束详解就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持三水点靠木。

Python 相关文章推荐
关于你不想知道的所有Python3 unicode特性
Nov 28 Python
通过C++学习Python
Jan 20 Python
python在windows命令行下输出彩色文字的方法
Mar 19 Python
全面解析Python的While循环语句的使用方法
Oct 13 Python
python实现多线程抓取知乎用户
Dec 12 Python
基于Django模板中的数字自增(详解)
Sep 05 Python
Python实现爬取百度贴吧帖子所有楼层图片的爬虫示例
Apr 26 Python
selenium3+python3环境搭建教程图解
Dec 07 Python
Python定义函数功能与用法实例详解
Apr 08 Python
python基于递归解决背包问题详解
Jul 03 Python
python和JavaScript哪个容易上手
Jun 23 Python
Keras在mnist上的CNN实践,并且自定义loss函数曲线图操作
May 25 Python
让Django的BooleanField支持字符串形式的输入方式
May 20 #Python
django 解决model中类写不到数据库中,数据库无此字段的问题
May 20 #Python
基于Python脚本实现邮件报警功能
May 20 #Python
完美解决Django2.0中models下的ForeignKey()问题
May 19 #Python
Django 解决model 反向引用中的related_name问题
May 19 #Python
django queryset 去重 .distinct()说明
May 19 #Python
django正续或者倒序查库实例
May 19 #Python
You might like
PHP的类 功能齐全的发送邮件类
2006/10/09 PHP
用PHP读取超大文件的实例代码
2012/04/01 PHP
PHP解压ZIP文件到指定文件夹的方法
2016/11/17 PHP
DIY jquery plugin - tabs标签切换实现代码
2010/12/11 Javascript
JavaScript 实现类的多种方法实例
2013/05/01 Javascript
jquery如何把数组变为字符串传到服务端并处理
2014/04/30 Javascript
jQuery使用$.get()方法从服务器文件载入数据实例
2015/03/25 Javascript
用JavaScript来美化HTML的select标签的下拉列表效果
2015/11/17 Javascript
跟我学习javascript的this关键字
2020/05/28 Javascript
AngularJS入门教程之数据绑定用法示例
2016/11/01 Javascript
npm国内镜像 安装失败的几种解决方案
2017/06/04 Javascript
在vue中实现简单页面逆传值的方法
2017/11/27 Javascript
jQuery创建及操作xml格式数据示例
2018/05/26 jQuery
webpack4 + react 搭建多页面应用示例
2018/08/03 Javascript
Vue项目安装插件并保存
2019/01/28 Javascript
jQuery实现高级检索功能
2019/05/28 jQuery
NodeJs 模仿SIP话机注册的方法
2019/06/21 NodeJs
python字符串的常用操作方法小结
2016/05/21 Python
python bottle框架支持jquery ajax的RESTful风格的PUT和DELETE方法
2017/05/24 Python
编写多线程Python服务器 最适合基础
2018/09/14 Python
python cumsum函数的具体使用
2019/07/29 Python
Python读入mnist二进制图像文件并显示实例
2020/04/24 Python
利用Python实现字幕挂载(把字幕文件与视频合并)思路详解
2020/10/21 Python
45个非常奇妙的CSS3 特性应用示例
2012/01/01 HTML / CSS
日本最大的眼镜购物网站:Oh My Glasses
2016/11/13 全球购物
最新茶叶店创业计划书
2014/01/14 职场文书
校园文明倡议书
2014/05/16 职场文书
授权委托书公证
2014/09/14 职场文书
党的群众路线教育实践活动总结
2014/10/30 职场文书
2014年禁毒工作总结
2014/11/24 职场文书
三行辞职书范文
2015/02/26 职场文书
朋友聚会开场白
2015/06/01 职场文书
Redis实现订单自动过期功能的示例代码
2021/05/08 Redis
python plt.plot bar 如何设置绘图尺寸大小
2021/06/01 Python
Redis 操作多个数据库的配置的方法实现
2022/03/23 Redis
windows server 2012安装FTP并配置被动模式指定开放端口
2022/06/10 Servers