【问题标题】:Rendering ForeignKey objects in template with Django使用 Django 在模板中渲染 ForeignKey 对象
【发布时间】:2020-12-18 10:18:31
【问题描述】:

我无法在模板中呈现相关对象。我有一个 Page 模型,它与自身有一个 ForeignKey 关系来定义父子关系。

class Page(BaseModel):
    title = models.CharField(
        max_length=280,
        blank=False,
        null=False,
    )
    description = models.CharField(
        max_length=280,
        blank=False,
        null=False,
    )
    slug = models.SlugField(
        blank=False,
        null=False,
    )
    is_home = models.BooleanField(
        default=False,
    )
    is_parent = models.BooleanField(
        default=False,
    )
    parent = models.ForeignKey(
        'self',
        on_delete=models.PROTECT,
        default='Home',
        blank=True,
        null=True,
        related_name='children',
        )
    content = RichTextField(
        blank=False,
        null=False,
    )

    def __str__(self):
        return self.title

    def get_absolute_url(self):
        return reverse('page_detail', args=[str(self.slug)])

我的views.py 过滤掉了非父页面:

class PageListView(ListView):
    queryset = Page.objects.filter(is_parent=True, is_home=False)
    template_name = 'pages/page_list.html'
    context_object_name = 'page'

但是在我的模板中渲染“子”对象时,我被卡住了。我认为我需要一个循环内的循环(第一个用于父页面,第二个用于每个父页面的子页面),但无法让第二个循环工作。这是我最近的尝试,它得到了一个“'Page' object is not iterable”错误。

{% extends '_base.html' %}

{% block content %}
<div class="container">
    {% for page in page %}
        <p>{{ page.title }}</p>
           {% for children in page %}
              <p>{{ page.title }}</p>
           {% endfor %}
    {% endfor %}
</div>
{% endblock content %}

【问题讨论】:

    标签: django django-models django-views django-templates django-queryset


    【解决方案1】:

    你可以遍历由related_name构造的管理器:

    {% block content %}
    <div class="container">
        {% for page in page %}
            <p>{{ page.title }}</p>
               {% for child in page.children.all %}
                  <p>{{ child.title }}</p>
               {% endfor %}
        {% endfor %}
    </div>
    {% endblock content %}

    您可以使用.prefetch_related(…) [Django-doc] 子句提高效率:

    class PageListView(ListView):
        queryset = Page.objects.filter(
            is_parent=True, is_home=False
        ).prefetch_related('children')
        template_name = 'pages/page_list.html'
        context_object_name = 'page'

    我还建议不要创建一个额外的字段is_parent,因为这是一种数据复制。事实证明,即使在同一个数据库上,保持字段同步也可能比乍看之下更难。您可以检查对象是否为父对象:

    class PageListView(ListView):
        queryset = Page.objects.filter(
            children__isnull=False, is_home=False
        ).distinct().prefetch_related('children')
        template_name = 'pages/page_list.html'
        context_object_name = 'page'

    【讨论】:

    • 感谢@Willem-van-onsem - 问题已解决!还将听取您的建议,以确保我的查询也有效。
    猜你喜欢
    • 1970-01-01
    • 2018-11-25
    • 2022-07-22
    • 1970-01-01
    • 2017-12-29
    • 2011-09-28
    • 2012-06-25
    • 1970-01-01
    • 2018-05-20
    相关资源
    最近更新 更多