一、一对多关系: ForeignKey

一对多是最常见的模型关系,例如 "作者 - 书籍" 场景:假设一个作者可以写多本书,但每本书只能属于一个作者。

定义关系

核心参数说明:

  • on_delete=models.CASCADE:当作者被删除时,关联的书籍也会被自动删除
  • related_name='books':定义反向查询名称,可通过author.books.all()获取作者的所有书籍
from django.db import models

class Author(models.Model):
first_name = models.CharField(max_length=100)
last_name = models.CharField(max_length=100) def __str__(self):
return f"{self.first_name} {self.last_name}" class Book(models.Model):
title = models.CharField(max_length=100)
publication_date = models.DateField()
# 外键关联Author,级联删除,反向查询名为books
author = models.ForeignKey(
Author,
on_delete=models.CASCADE,
related_name='books'
) def __str__(self):
return self.title

数据操作示例

创建数据

# 创建作者
author1 = Author.objects.create(first_name='J.K.', last_name='Rowling')
author2 = Author.objects.create(first_name='George', last_name='Orwell') # 创建书籍并关联作者
book1 = Book.objects.create(
title='Harry Potter',
publication_date='1997-06-26',
author=author1
)
book2 = Book.objects.create(
title='1984',
publication_date='1949-06-08',
author=author2
)

查询操作

# 正向查询:通过书籍找作者
book = Book.objects.get(title='1984')
print(book.author) # 输出: George Orwell # 反向查询:通过作者找书籍
author = Author.objects.get(last_name='Rowling')
for book in author.books.all():
print(book.title) # 输出: Harry Potter

高级配置

禁用外键约束:当需要灵活管理关联关系(如允许删除存在关联数据的主表记录)时,可关闭数据库级约束

author = models.ForeignKey(
Author,
on_delete=models.SET_NULL,
related_name='books',
db_constraint=False, # 不创建数据库外键约束
null=True
)

自定义数据库列名:默认会生成<ClassName>_id列,可通过db_column修改

dept_id = models.ForeignKey(
"SystemDept",
on_delete=models.SET_NULL,
db_column="dept_id", # 显式指定数据库列名
null=True
)

二、多对多关系: ManyToManyField

多对多关系适用于 "作者 - 书籍" 的另一种场景:假设一个作者可以写多本书,一本书也可以有多个作者。

定义关系

Django 会自动创建中间表(默认名为appname_book_authors)存储关联关系,无需手动定义。

from django.db import models

class Author(models.Model):
name = models.CharField(max_length=100)
email = models.EmailField() def __str__(self):
return self.name class Book(models.Model):
title = models.CharField(max_length=200)
publication_date = models.DateField()
# 多对多关联Author
authors = models.ManyToManyField(Author, related_name='books') def __str__(self):
return self.title

数据操作示例

添加 / 移除关联

# 创建实例
author1 = Author.objects.create(name='Alice', email='alice@example.com')
author2 = Author.objects.create(name='Bob', email='bob@example.com')
book = Book.objects.create(title='Example Book', publication_date='2023-01-01') # 添加关联
book.authors.add(author1, author2) # 移除关联
book.authors.remove(author1)

查询操作

# 正向查询:书籍的所有作者
book = Book.objects.get(title='Example Book')
for author in book.authors.all():
print(author.name) # 反向查询:作者的所有书籍
author = Author.objects.get(name='Bob')
for book in author.books.all(): # related_name='books'
print(book.title)

自定义中间表

当需要存储关联关系的额外信息(如邀请原因、加入时间)时,可自定义中间表

class Membership(models.Model):
group = models.ForeignKey(Group, on_delete=models.CASCADE)
person = models.ForeignKey(Person, on_delete=models.CASCADE)
inviter = models.ForeignKey(Person, related_name="invites", on_delete=models.CASCADE)
invite_reason = models.CharField(max_length=64) # 额外信息 class Group(models.Model):
name = models.CharField(max_length=128)
members = models.ManyToManyField(
Person,
through="Membership", # 指定中间表
through_fields=("group", "person"), # 关联字段
)

三、性能优化技巧

select_related:用于一对多关系,提前加载关联对象,减少数据库查询

# 普通查询(N+1问题)
entries = Entry.objects.all()
for entry in entries:
print(entry.blog.name) # 每次循环都会触发新查询 # 优化后(仅1次查询)
entries = Entry.objects.select_related('blog').all()
for entry in entries:
print(entry.blog.name) # 使用缓存数据

批量操作:利用update()进行批量更新,避免循环操作

# 批量标记站内信为已读
SystemNotifyMessage.objects.filter(
id__in=ids.split(",")
).update(
read_status=True,
read_time=timezone.now()
)

四、关于是否使用外键约束

在实际项目中,是否使用数据库外键约束需要权衡利弊

使用外键的优势

  • 数据完整性:数据库级别的约束保证关联数据一致性
  • 开发效率:ORM 自动处理关联查询和级联操作
  • 查询便捷:支持select_related等优化方法,简化多表查询

禁用外键的场景

  • 高并发系统:外键会增加数据库锁竞争,影响写入性能
  • 分布式架构:分库分表环境下,跨库外键无法生效
  • 复杂迁移:避免循环依赖导致的迁移失败问题

折中方案:使用db_constraint=False 参数

  • 数据库层面:无外键约束,数据库不会强制校验关联数据的存在性
  • Django ORM 层面:保留逻辑关联,ORM仍将字段视为外键关系(逻辑关联),支持 ORM 查询、操作语法
特性 db_constraint=True (默认) db_constraint=False
数据库外键约束 创建,强制数据一致性 不创建
级联操作 数据库自动处理 仅由 Django ORM 处理
关联数据存在性校验 数据库强制校验 不校验(需应用层保障)
ORM 查询支持 完整支持 完整支持(逻辑外键保留)
性能影响 外键约束带来额外开销 无约束开销
适用场景 强数据一致性需求 高频写入/跨库/历史数据迁移

五、多对多关系实战

实战场景:在一个后台管理系统中,用户与角色往往是多对多关系。一个用户可以分配多个角色,一个角色也可以属于多个用户。

模型定义:点击查看完整代码

class SystemUsers(BaseModel, AbstractBaseUser):
id = models.BigAutoField(primary_key=True, db_comment="用户ID", help_text="用户ID")
username = models.CharField(
max_length=30, unique=True, db_comment="用户账号", help_text="用户账号"
)
# ...
# 与角色多对多关系
roles = models.ManyToManyField(
"SystemRole",
through="SystemUserRole",
through_fields=("user_id", "role_id"),
related_name="users",
)
# ... class SystemUserRole(BaseModel):
"""用户和角色关联中间表"""
id = models.BigAutoField(primary_key=True, db_comment="id")
user_id = models.ForeignKey(
"SystemUsers",
on_delete=models.CASCADE,
db_constraint=False,
db_column="user_id",
db_comment="用户ID",
)
role_id = models.ForeignKey(
"SystemRole",
on_delete=models.CASCADE,
db_constraint=False,
db_column="role_id",
db_comment="角色ID",
) class Meta:
managed = True
db_table = "system_user_role"
db_table_comment = "用户和角色关联表"
ordering = ["-id"]

system_user_role数据库生成的中间表


您正在阅读的是《Django从入门到实战》专栏!关注不迷路~

Django模型关系:从一对多到多对多全解析的更多相关文章

  1. Python进阶----表与表之间的关系(一对一,一对多,多对多),增删改查操作

    Python进阶----表与表之间的关系(一对一,一对多,多对多),增删改查操作,单表查询,多表查询 一丶表与表之间的关系 背景: ​ ​ ​  ​ ​ 由于如果只使用一张表存储所有的数据,就会操作数 ...

  2. django 模型关系

    模型关系 关系数据库的威力体现在表之间的相互关联,Django提供了三种最常见的数据库关系:多对一 (many-to-one),多对多(many-to-many),一对一(one-to-one) 多对 ...

  3. Nhibernate 映射关系,一对多 多对一与多对手在映射文件中的体现。

    今天做了第一个Nhibernate项目,摸着石头过河,学到了一些东西,在这里将自己总结体会到的一些映射关系写出来,与大家分享,由于是初学者,如果有不对的地方希望大家能够指出来. 首先要说明要建立的几张 ...

  4. Django模型层—ORM

    目录 一.模型层(models) 1-1. 常用的字段类型 1-2. 字段参数 1-3. 自定义char字段 1-4. 外键关系 二.Django中测试脚本的使用 三.单表操作 3-1. 添加记录 3 ...

  5. 八.django模型系统(二)之常用查询及表关系的实现

    Ⅰ.常用查询  1.几个概念 每一个django模型类,都有一个默认的管理器,objects,查询就是依赖于objects管理器进行的(在创建时就被添加了). QuerySet表示数据库中对象的列表( ...

  6. hibernate中一对多多对一关系设计的理解

    1.单向多对一和双向多对一的区别? 只需要从一方获取另一方的数据时 就使用单向关联双方都需要获取对方数据时 就使用双向关系 部门--人员 使用人员时如果只需要获取对应部门信息(user.getdept ...

  7. Django 08 Django模型基础3(关系表的数据操作、表关联对象的访问、多表查询、聚合、分组、F、Q查询)

    Django 08 Django模型基础3(关系表的数据操作.表关联对象的访问.多表查询.聚合.分组.F.Q查询) 一.关系表的数据操作 #为了能方便学习,我们进入项目的idle中去执行我们的操作,通 ...

  8. django模型层之多表关系

    一. 多表操作 数据库表关系之关联字段与外键约束 一对多 book(多) publish(一) 查询<<水浒传>>这本书出版社的地址: select publish_id fr ...

  9. django模型中有外键关系的表删除相关设置

    0904自我总结 django模型中有外键关系的表删除相关设置 一.一对一 例如有Author.AuthorDetail两表 author = models.OneToOneField(to='Aut ...

  10. django模型中的关系对应

    显然,关系数据库的力量在于将表相互关联.Django提供了定义三种最常见的数据库关系类型的方法:多对一,多对多和一对一. 在说明之前,首先来理解一下这三个概念: 多对一: 两个集合a,b;集合a中的多 ...

随机推荐

  1. C++ 容器(STL容器)

    容器(container)用于存放数据的类模板.可变长数组.链表.平衡二叉树等数据结构在STL中都被实现为容器. 在使用容器时,即将容器类模型实例化为容器类,会指明容器中存放的元素是什么类型. 容器可 ...

  2. uni-app项目从0-1基础架构搭建全流程

    前情 最近新接了一个全新项目,我负责从0开始搭建小程序,我选用的技术栈是uni-app技术栈,UI库选择的是uview-plus,CSS引入现在流行的tainlwindcss,实现CSS原子化书写,实 ...

  3. Windows平台调试器原理与编写01.调试框架

    调试器原理与编写01.调试框架-C/C++基础-断点社区-专业的老牌游戏安全技术交流社区 - BpSend.net 调试框架 调试器最基本功能: 断点,单步 断点分为三类 软件断点 硬件断点 内存断点 ...

  4. UVW源码漫谈(一)

    博客园是个非常好的学习知识的地方,相信有很多人跟我一样,园龄3年,从博客园不知道拷了多少代码,看了多少博客,自己却一篇博客都没写过.真是罪过. 这次准备写几篇关于这个项目源码的阅读和理解的文章,大家一 ...

  5. Spring注解之@Autowired:装配构造函数和属性

    在User类中创建一个构造函数,传入参数student: import org.springframework.beans.factory.annotation.Autowired; import o ...

  6. Ceph对象存储集群常用操作

    一.常用操作 1.1 查看某个bucket详情 # 查看bucket列表 [root@oss001 ~]# radosgw-admin bucket list [ "awss3e68c588 ...

  7. ChatMoney化身恋爱大师来帮助你收获完美爱情!

    本文由 ChatMoney团队出品 介绍说明 在纷繁复杂的情感世界里,寻找真爱的道路常常充满迷茫和困惑.但现在,有了 AI 智能体恋爱大师,为您的爱情之旅点亮明灯. AI 智能体恋爱大师并非传统意义上 ...

  8. 分布式可视化 DAG 任务调度系统 Taier 的整体流程分析

    Taier 作为袋鼠云的开源项目之一,是一个分布式可视化的 DAG 任务调度系统.旨在降低 ETL 开发成本,提高大数据平台稳定性,让大数据开发人员可以在 Taier 直接进行业务逻辑的开发,而不用关 ...

  9. Linux 查找Redis配置信息

    前言 有时在使用Redis时密码或者配置信息经常忘记,应该怎么找回呢? 解决 如果设置了自启动,先查找服务状态(systemctl status redis服务名称) 根据服务可以找到服务的启动脚本, ...

  10. error Delete `␍`eslintprettier/prettier

    问题描述 今天提交代码的时候,控制台报的错误就有点吓人 分析问题 由于历史原因,windows下和linux(mac)下的文本文件的换行符不一致. 当我用windows电脑git操作代码的时候,如cl ...