0

使用 aGenericRelation将 s 映射RecordPersons,我有以下代码,它可以正常工作,但是我正在尝试解决一个性能问题:

模型.py

class RecordX(Record):  # Child class
    ....

class Record(models.Model):  # Parent class
    people = GenericRelation(PersonRecordMapping)

    def people_all(self):
        # Use select_related here to minimize the # of queries later
        return self.people.select_related('person').all()

    class Meta:
        abstract = True

class PersonRecordMapping(models.Model):
    person = models.ForeignKey(Person)
    content_type = models.ForeignKey(ContentType)
    object_id = models.PositiveIntegerField()
    content_object = GenericForeignKey('content_type', 'object_id')

class Person(models.Model):
    ...

总之,我有:

RecordX ===GenericRelation===> PersonRecordMapping ===ForeignKey===> Person

我的应用程序中的显示逻辑遵循这些关系来显示Person映射到每个的所有 s RecordX

视图.py

@rendered_with('template.html')
def show_all_recordXs(request):
    recordXs = RecordX.objects.all()
    return { 'recordXs': recordXs }

问题出在模板中:

模板.html

{% for recordX in recordXs %}
    {# Display RecordX's other fields here #}
    <ul>
    {% for map in record.people_all %}{# <=== This generates a query every time! #}
        <li>{{ map.person.name }}</li>
    {% endfor %}
    </ul>
{% endfor %}

如前所述,每次我请求与Persons 相关的 s 时RecordX,它都会生成一个新查询。我似乎无法弄清楚最初如何预取这些以避免冗余查询。

如果我尝试selected_related,我会得到一个错误,selected_related这里没有字段是可能的(错误消息:)Invalid field name(s) given in select_related: 'xxx'. Choices are: (none)。毫不奇怪,我现在明白了——这个模型上没有任何 FK。

如果我尝试prefetch_related('people'),不会引发错误,但我仍然会在模板中的每个循环上得到与以前相同的重复查询。同样对于prefetch_related('people__person'). 如果我尝试prefetch_related('personrecordmapping'),我会得到一个错误。

按照这个答案,我考虑过尝试模拟一个类似的select_related东西:

PersonRecordMapping.objects.filter(object_id__in=RecordX.objects.values_list('id', flat=True))

但我不太了解_content_object_cache如何使这个答案适应我的情况(即使是正确的方法)。

那么 -当页面将显示n 个对象时,如何将所有Persons预取到RecordXs 以避免n 个查询? RecordX

谢谢你的帮助!

4

1 回答 1

0

Ack,显然我需要同时调用这两个-- prefetch_related('people', 'people__person')。叹。

于 2017-01-17T05:20:36.657 回答