【Django】执行查询—检索对象

检索对象

以下述模型为基础,讨论检索对象的方式方法:

python 复制代码
from datetime import date

from django.db import models


class Blog(models.Model):
    name = models.CharField(max_length=100)
    tagline = models.TextField()

    def __str__(self):
        return self.name


class Author(models.Model):
    name = models.CharField(max_length=200)
    email = models.EmailField()

    def __str__(self):
        return self.name


class Entry(models.Model):
    blog = models.ForeignKey(Blog, on_delete=models.CASCADE)
    headline = models.CharField(max_length=255)
    body_text = models.TextField()
    pub_date = models.DateField()
    mod_date = models.DateField(default=date.today)
    authors = models.ManyToManyField(Author)
    number_of_comments = models.IntegerField(default=0)
    number_of_pingbacks = models.IntegerField(default=0)
    rating = models.IntegerField(default=5)

    def __str__(self):
        return self.headline

检索全部对象

python 复制代码
>>> all_entries = Entry.objects.all()

通过过滤器检索指定对象

通过添加过滤条件精炼原始 QuerySet。两种最常见的精炼 QuerySet 的方式是:
filter(**kwargs)

返回一个新的 QuerySet,包含的对象满足给定查询参数。
exclude(**kwargs)

返回一个新的 QuerySet,包含的对象 不 满足给定查询参数。

由此可见,对 QuerySet 进行细化的结果本身也是一个 QuerySet,因此可以将细化操作链接在一起。

QuerySets 是惰性的 ------ 创建 QuerySet 的过程不涉及任何数据库活动。你可以一直堆叠过滤条件,但 Django 实际上不会运行查询,直到 QuerySet 被评估

检索单个对象

python 复制代码
>>> one_entry = Entry.objects.get(pk=1)

QuerySet条目数限制

QuerySet是一个对象列表,因此类似python数组切片,可以对条目数进行限制,但是也有区别,不支持负索引。

通常对 QuerySet 进行切片会返回一个新的 QuerySet,它不会评估查询。但是如果使用了 Python 切片语法的 "step" 参数,会实际执行查询。

例子:

python 复制代码
# 返回前5个对象(对应SQL:LIMIT 5)
>>> Entry.objects.all()[:5]
# 返回第六到第十个对象(对应SQL:OFFSET 5 LIMIT 5)
>>> Entry.objects.all()[5:10]

# 不支持负索引,如 Entry.objects.all()[-1]

# 返回前10个对象中的第2、4、6、8、10个对象
>>> Entry.objects.all()[:10:2]

字段查询

字段查询即你如何制定 SQL WHERE 子句。它们以关键字参数的形式传递给 QuerySet 方法 filter(), exclude() 和 get()。

基本的查找关键字参数采用形式 field__lookuptype=value (使用双下划线,前面是字段名,后面是查询方式参数,查询ForeignKey时可以使用外键字段名_id)。

例子:

python 复制代码
>>> Entry.objects.filter(pub_date__lte="2006-01-01")

查询方式参数有:exact,iexact,contains,icontains,startswith,endswith等等

跨关系查询

为了跨越关系,跨模型使用关联字段名,字段名由双下划线分割,直到拿到想要的字段。

例子:

python 复制代码
# 检索所有具有 name 为 'Beatles Blog' 的 Blog 的 Entry 对象
>>> Entry.objects.filter(blog__name="Beatles Blog")

也可反向查询:

python 复制代码
# 检索所有至少有一个 headline 包含 'Lennon' 的 Entry 的 Blog 对象
>>> Blog.objects.filter(entry__headline__contains="Lennon")
相关推荐
彭泽布衣1 小时前
python2.7/lib-dynload/_ssl.so: undefined symbol: sk_pop_free
python·sk_pop_free
头发那是一根不剩了1 小时前
Spring Boot 多数据源切换:AbstractRoutingDataSource
数据库·spring boot·后端
喜欢吃豆1 小时前
从零构建MCP服务器:FastMCP实战指南
运维·服务器·人工智能·python·大模型·mcp
草履虫建模2 小时前
Redis:高性能内存数据库与缓存利器
java·数据库·spring boot·redis·分布式·mysql·缓存
一个处女座的测试2 小时前
Python语言+pytest框架+allure报告+log日志+yaml文件+mysql断言实现接口自动化框架
python·mysql·pytest
nananaij2 小时前
【Python基础入门 re模块实现正则表达式操作】
开发语言·python·正则表达式
蛋仔聊测试2 小时前
Playwright 网络流量监控与修改指南
python
泊浮目2 小时前
未来数据库硬件-计算篇
数据库·云计算·操作系统
靖顺2 小时前
【OceanBase诊断调优】—— 执行计划显示分区 PARTITIONS[P0SP9] 如何查询是哪个分区?
数据库·oracle·oceanbase
小王子10243 小时前
Django+DRF 实战:序列化器 ValidationError 触发机制完整指南
前端·django