【发布时间】:2017-03-16 15:13:48
【问题描述】:
我在具有 5 个 m2m 字段的模型上遇到了 prefetch_related 的一些严重性能问题,并且我还预取了几个嵌套的 m2m 字段。
class TaskModelManager(models.Manager):
def get_queryset(self):
return super(TaskModelManager, self).get_queryset().exclude(internalStatus=2).prefetch_related("parent", "takes", "takes__flags", "assignedUser", "assignedUser__flags", "asset", "asset__flags", "status", "approvalWorkflow", "viewers", "requires", "linkedTasks", "activities")
class Task(models.Model):
uuid = models.UUIDField(primary_key=True, default=genOptimUUID, editable=False)
internalStatus = models.IntegerField(default=0)
parent = models.ForeignKey("self", blank=True, null=True, related_name="childs")
name = models.CharField(max_length=45)
taskType = models.ForeignKey("TaskType", null=True)
priority = models.IntegerField()
startDate = models.DateTimeField()
endDate = models.DateTimeField()
status = models.ForeignKey("ProgressionStatus")
assignedUser = models.ForeignKey("Asset", related_name="tasksAssigned")
asset = models.ForeignKey("Asset", related_name="tasksSubject")
viewers = models.ManyToManyField("Asset", blank=True, related_name="followedTasks")
step = models.ForeignKey("Step", blank=True, null=True, related_name="tasks")
approvalWorkflow = models.ForeignKey("ApprovalWorkflow")
linkedTasks = models.ManyToManyField("self", symmetrical=False, blank=True, related_name="linkedTo")
requires = models.ManyToManyField("self", symmetrical=False, blank=True, related_name="depends")
objects = TaskModelManager()
查询次数很好,数据库查询时间也很好,例如,如果我查询模型的 700 个对象,我有 35 个查询,平均查询时间为 100~200 毫秒,但总请求时间约为 8 秒.
我进行了一些分析,它指出超过 80% 的时间都花在了 prefetch_related_objects 调用上。
我正在使用Django==1.8.5 和djangorestframework==3.4.6
我愿意以任何方式优化它。 提前感谢您的帮助。
使用select_related编辑:
我已经尝试过 Alasdair 提出的改进
class TaskModelManager(models.Manager):
def get_queryset(self):
return super(TaskModelManager, self).get_queryset().exclude(internalStatus=2).select_related("parent", "status", "approvalWorkflow", "step").prefetch_related("takes", "takes__flags", "assignedUser", "assignedUser__flags", "asset", "asset__flags", "viewers", "requires", "linkedTasks", "activities")
对于 32 次查询和 150 毫秒查询时间的请求,新结果仍然是 8 秒。
编辑:
似乎 4 年前在 Django 问题跟踪器上打开了一张票,并且仍然打开:https://code.djangoproject.com/ticket/20577
【问题讨论】:
标签: python django postgresql prefetch