我经常发现自己想从Django的查询集中获取第一个对象,或者如果没有的话返回None
。有很多方法可以做到这一点,但我想知道哪一个是最好的。
qs = MyModel.objects.filter(blah = blah)
if qs.count() > 0:
return qs[0]
else:
return None
这会导致两次数据库调用吗?这似乎是浪费。这样会更快吗?
qs = MyModel.objects.filter(blah = blah)
if len(qs) > 0:
return qs[0]
else:
return None
另一种选择是:
qs = MyModel.objects.filter(blah = blah)
try:
return qs[0]
except IndexError:
return None
这会生成一个数据库调用,这很好,但是需要创建一个异常对象,这是一件非常占用内存的事情,因为你真正需要的只是一个简单的if测试。
我怎样才能只用一次数据库调用,而不用异常对象搅动内存来完成这一任务呢?
9条答案
按热度按时间sigwle7e1#
Django 1.6 (released Nov 2013)引入了方便的方法
first()
和last()
,如果查询集没有返回对象,则它们会吞下产生的异常并返回None
。vpfxa7rd2#
可以使用数组切片:
可与
.filter()
一起使用:你不想先把它变成一个列表,因为这会强制调用所有记录的完整数据库。只要做上面的操作,它只会拉取第一个。你甚至可以使用
.order_by()
来确保你得到你想要的第一个。确保添加
.get()
,否则您将得到一个QuerySet,而不是一个对象。9gm1akwq3#
现在,在Django1.9中你有
first()
方法来处理查询集。这是比
.get()
或[0]
更好的方法,因为如果queryset为空,它不会抛出异常。因此,您不需要使用exists()
进行检查dbf7pr2w4#
zengzsys5#
这也可以起作用:
如果为空,则返回空列表,否则返回列表中的第一个元素。
iih3973s6#
如果您计划经常获取第一个元素-您可以在这个方向上扩展QuerySet:
按如下方式定义模型:
并像这样使用它:
dzjeubhm7#
可以这样
或
tvokkenx8#
你应该使用django的方法,比如exists,它就在那里等着你去使用。
jgwigjjp9#
我发现了一些东西,我认为可能会对任何人谁碰到这个页面有帮助。
**当使用prefetch_related()**时,调用
.first()
实际上会对数据库进行另一次查询(使用一些OFFSET
和LIMIT
子句),而.all()
只返回查询集已经预取的数据。考虑到这一点,当使用
prefetch_related
从Django的查询集中选取第一项时,我会选择这个方法: