Django双下划线查询

在 Django 中,双下划线(__)查询是一种强大的语法,用于在 ORM(对象关系映射)中跨关联模型进行字段查询,或使用内置查询表达式(如过滤、聚合等)。以下是其核心用法和示例:

1. 跨关联模型查询

通过双下划线可以引用关联模型(ForeignKey、ManyToManyField、OneToOneField)的字段。

示例模型:

复制代码
class Author(models.Model):
    name = models.CharField(max_length=100)

class Book(models.Model):
    title = models.CharField(max_length=100)
    author = models.ForeignKey(Author, on_delete=models.CASCADE)

查询示例:

复制代码
# 查询作者名为"Alice"的所有书籍
books = Book.objects.filter(author__name="Alice")

# 查询所有写过标题包含"Python"的书籍的作者
authors = Author.objects.filter(book__title__icontains="Python")

2. 字段查找表达式

双下划线后可接多种查询表达式,如:
比较运算符

exact:精确匹配(默认)

gt、gte、lt、lte:大于、大于等于、小于、小于等于

in:包含在列表中

range:范围查询

示例:

复制代码
# 查询价格大于100的产品
products = Product.objects.filter(price__gt=100)

# 查询ID为1、3、5的用户
users = User.objects.filter(id__in=[1, 3, 5])

字符串处理

contains、icontains:包含(区分 / 不区分大小写)

startswith、istartswith:以... 开头

endswith、iendswith:以... 结尾

regex、iregex:正则表达式匹配

示例:

复制代码
# 查询邮箱以"@gmail.com"结尾的用户
users = User.objects.filter(email__endswith="@gmail.com")

日期和时间处理

year、month、day:提取日期部分

hour、minute、second:提取时间部分

date:日期比较

time:时间比较

示例:

复制代码
# 查询2023年创建的订单
orders = Order.objects.filter(created_at__year=2023)

# 查询昨天创建的所有对象
from datetime import date, timedelta
yesterday = date.today() - timedelta(days=1)
objects = MyModel.objects.filter(created_at__date=yesterday)

空值和布尔值

isnull:是否为空(True/False)

exact:布尔值匹配

示例:

复制代码
# 查询没有关联图片的文章
articles = Article.objects.filter(image__isnull=True)

# 查询已发布的文章
articles = Article.objects.filter(is_published__exact=True)

3. 多级跨关联查询

可以通过多个双下划线连接多层关联模型。

示例模型:

复制代码
class Country(models.Model):
    name = models.CharField(max_length=100)

class City(models.Model):
    name = models.CharField(max_length=100)
    country = models.ForeignKey(Country, on_delete=models.CASCADE)

class User(models.Model):
    name = models.CharField(max_length=100)
    city = models.ForeignKey(City, on_delete=models.CASCADE)

查询示例:

复制代码
# 查询来自中国北京的所有用户
users = User.objects.filter(city__name="Beijing", city__country__name="China")

4. 聚合查询与双下划线

在聚合函数(如Sum、Avg)中也可使用双下划线跨模型引用字段。

示例:

复制代码
from django.db.models import Sum
# 计算每个作者的总书籍销量
authors = Author.objects.annotate(total_sales=Sum("book__sales"))

注意事项

性能问题:多级跨关联可能导致 SQL 查询复杂度增加,建议结合select_related()或prefetch_related()优化。

反向关联:通过关联模型反向查询时,默认使用模型名小写加_set(如author.book_set.all()),但在双下划线查询中可直接使用模型名小写(如author__name)。

空值处理:使用跨关联查询时,若关联对象不存在,查询结果可能为空,需谨慎处理。

通过双下划线查询,Django ORM 可以简洁地表达复杂的数据库查询逻辑,避免编写原生 SQL。

相关推荐
ClouGence26 分钟前
Oracle 数据同步为什么会出现数据不一致?长事务是常被忽略的原因
数据库·后端·oracle
飞将3 小时前
从零实现数据库(2)——HashIndex + IndexManager
数据库
兵慌码乱13 小时前
面向桌面端的资产管理系统分层架构设计与核心模块实现
python·系统架构·sqlite·pyqt5·数据库设计·桌面应用开发·mvc架构
呱呱复呱呱1 天前
Django CBV 源码解读:一个请求是怎么找到你的 get() 方法的
python·django
Nturmoils1 天前
订单列表慢查询,先看 WHERE、ORDER BY 和 LIMIT
数据库
渣波1 天前
拒绝 SQL 焦虑!手把手带你用 NestJS + Prisma + DTO 写出“防弹”级后端代码
javascript·数据库·后端
兵慌码乱2 天前
基于Python+PyQt5+SQLite的药房管理系统实现:事务一致性与界面解耦全流程解析
python·sqlite·信号与槽·pyqt5·数据库设计·桌面应用开发·事务处理
倔强的石头_2 天前
KingbaseES 新版MySQL 兼容版体验:旧版迁移 + 功能实测
数据库
倔强的石头_5 天前
《Kingbase护城河》——数据库存储空间全景探测与精细化瘦身实战
数据库
冬奇Lab6 天前
每日一个开源项目(第134篇):Zvec - 阿里开源的嵌入式向量数据库,向量搜索界的 SQLite
数据库·人工智能·llm