Python中Django模板渲染嵌套的生成器表达式的问题

views 部分复现代码

from django.http import HttpResponse
from django.template import Context
from django.template import Template

def test(request): src = [ {“a”: 1, “b”: 2}, {“a”: 10, “b”: 20}, ]

fields = ["a", "b"]
data = ((item[field] for field in fields) for item in src)

# for item in data:
#     print(list(item))  # 这样是正常的

template = Template('''
    {% for item in data %}
        {% for value in item %} {{ value }} {% endfor %}
    {% endfor %}
''')
context = Context({'data': data})
html = template.render(context)

print(html)
return HttpResponse()

模板里面内层生成器取到的 item 似乎是外层循环的最后一项,渲染结果都是 10 20。把内层生成器表达式改成列表推导式就正常了。很好奇这个问题的原因。


Python中Django模板渲染嵌套的生成器表达式的问题

3 回复

我遇到过这个问题。Django模板在处理嵌套的生成器表达式时确实会出问题,因为生成器只能迭代一次。

核心问题是:当你在模板里用嵌套循环时,内层生成器在第一次迭代后就被消耗完了,导致后续迭代无法获取数据。

看个具体例子:

# views.py
def my_view(request):
    # 外层是列表,内层是生成器表达式
    data = [
        (f"Group {i}", (x for x in range(3)))  # 内层是生成器
        for i in range(2)
    ]
    return render(request, 'template.html', {'data': data})
<!-- template.html -->
{% for group_name, items in data %}
    <h3>{{ group_name }}</h3>
    <ul>
    {% for item in items %}
        <li>{{ item }}</li>  <!-- 这里只会显示一次! -->
    {% endfor %}
    </ul>
{% endfor %}

解决方案:把生成器转成列表

最简单的修复方法就是在视图里把生成器转成列表:

# views.py - 修复版本
def my_view(request):
    data = [
        (f"Group {i}", list(x for x in range(3)))  # 关键:加上list()
        for i in range(2)
    ]
    return render(request, 'template.html', {'data': data})

或者用列表推导式:

data = [
    (f"Group {i}", [x for x in range(3)])  # 用列表推导式
    for i in range(2)
]

为什么这样?

Django模板的{% for %}标签在每次循环时都会重新迭代生成器,但生成器只能迭代一次。转成列表后,数据就被缓存了,可以多次访问。

一句话建议:在Django模板中使用嵌套数据时,确保内层数据是可重入的容器(如列表),而不是生成器。


测试的环境是 Python 3.7.2 和 Django 2.1.5

回到顶部