0

我简直被难住了。我正在尝试prefect_related在我的 django 应用程序中实现一个教科书案例,但它根本行不通。以下是相关型号:

class CanUseUnit(models.Model):

  objects = UnitManager()

  ingredient = models.ForeignKey('Ingredient', db_column='ingredient', related_name='useable_units')
  unit = models.ForeignKey('Unit', related_name='used_by', db_column='unit', limit_choices_to=models.Q(parent_unit__exact=None))

请注意,虽然这个模型有一个自定义的 Manager,但这是一个简单的子类,models.Managar只有 1 个方法 named get_all_info(),所以我认为这与我的问题没有任何关系:

我正在尝试查询所有成分,并预取它们的可用单位。这是我的 Django 查询:

def list_ingredients(request):
  ingredients = Ingredient.objects.all().order_by('accepted', 'name').prefetch_related('useable_units')
  for ingredient in ingredients:
    print(ingredient.useable_units.all())

  return render(request, 'admin/list_ingredients.html', {'ingredients': ingredients})

但是 Django 似乎在每条print语句上都命中了数据库……这是怎么回事?

编辑:

从模板调用时,缓存似乎确实有效。当我在上面的视图中省略打印语句时,访问包含以下模板代码的页面:

{% for ingredient in ingredients %}
<tr>
  <td><a href="/ingredients/edit/{{ ingredient.id }}/">{{ ingredient.name }}</a></td>
  <td align="center">{{ ingredient.useable_units.count }}</td>
</tr>

{% endfor %}

数据库只被击中两次(我预计,一次用于成分,一次用于 prefetch_related)。

已解决:好吧,问题似乎已经解决了……我一直在尝试并重新启动测试服务器,但突然它停止查询每种成分。不知道发生了什么,但我很高兴它解决了。

4

2 回答 2

1

我发现了问题。Shacki 的回答是错误的,与预取相关的确实只需要 2 个查询,而不是每个成分一个。

我的错误出现在模板的不同部分,我在其中显示了中间模型的属性。

所以我应该做的是以下几点:

ingredients = Ingredient.objects.all().order_by('accepted', 'name').prefetch_related('canuseunit_set')

代替:

ingredients = Ingredient.objects.all().order_by('accepted', 'name').prefetch_related('useable_units')
于 2013-10-21T11:03:39.897 回答
-2

如果您查看文档:https://docs.djangoproject.com/en/dev/ref/models/querysets/#prefetch-related,它会说

另一方面,prefetch_related 对每个关系进行单独的查找,并在 Python 中进行“加入”

据我了解,这正是你所看到的。

于 2013-07-30T15:36:38.567 回答