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

