4

我从我的数据库中获取多个具有预取关系的对象:

datei_logs = DateiLog.objects.filter(user=request.user)
                             .order_by("-pk")
                             .prefetch_related('transfer_logs')

transfer_logs指的是:

class TransferLog(models.Model):
  datei_log = models.ForeignKey("DateiLog", related_name="transfer_logs")
  status = models.CharField(
                    max_length=1,
                    choices=LOG_STATUS_CHOICES,
                    default='Good'
                    ) 
  server_name = models.CharField(max_length=100, blank=True, default="(no server)")
  server = models.ForeignKey('Server')

  class Meta:
    verbose_name_plural = "Transfer-Logs"

  def __unicode__(self):
    return self.server_name

现在我想获得所有TransferLogs状态为"Good". 但我想如果我这样做:

datei_logs[0].transfer_logs.filter(...)

它再次查询数据库!由于这发生在一个有许多日志条目的网站上,我最终得到了 900 个查询!

我用:

datei_logs[0].transfer_logs.count()

同样,它也会导致对数据库的大量查询!

我该怎么做才能“获取所有内容”,然后只查询一个包含所有信息而不是数据库的对象?

4

1 回答 1

4

由于您使用的是 Django 1.7,因此您可以使用新Prefetch()对象来指定要用于相关查找的查询集。

queryset = TransferLog.objects.filter(status='Good')
datei_logs = DateiLog.objects.filter(user=request.user)
                             .order_by("-pk")
                             .prefetch_related(Prefetch('transfer_logs', 
                                                        queryset=queryset,
                                                        to_attr='good_logs'))

然后您可以访问datei_logs[0].good_logs并检查len(datei_logs[0].good_logs).

如果您对多个状态感兴趣,则可以使用多个Prefetch对象。但是,如果您无论如何都要获取所有日志,那么您不妨坚持原来的查询,然后在 Python 中拆分日志,而不是调用filter().

于 2014-10-14T17:55:40.857 回答