我正在为我的Django应用程序构建一个csv导入表单,并希望在ModelFormSet
中显示要导入的行以用于验证目的。
因此,我向相关ModelAdmin
添加了一个视图,该视图从csv读取行并打印ModelFormSet(queryset=an_empty_queryset, initial={data_from_the_csv})
。
问题是模型通过ForeignKey
字段引用了其他三个模型,并且对于表单集中每个表单中的每个字段,都会发出数据库查询以填充{{1选项。
为什么Django不缓存表单(因为它被多次使用)或者是否已经有办法实现这一点我还不知道呢?
答案 0 :(得分:14)
Django formsets只是将表单创建的所有细节委托给表单对象本身,并且各个表单实例不知道其他表单实例,因此每个表单实例都必须查询自己的选择并不出乎意料。
缓存也可能会产生意想不到的副作用 - 例如,表单的__init__
函数可能依赖于它收到的initial
数据,从而导致缓存的form
对象不正确。
减少查询数量的最佳方法是检索选择查询集一次,然后将它们传递给构造函数中的表单类。这需要定义自定义ModelForm
和自定义ModelFormSet
。
您的表单需要一个直接接受选择的构造函数:
from django.forms.models import ModelForm
class MyForm(ModelForm):
def __init__(self, my_field_choices=None, *args, **kwargs):
super(MyForm, self).__init__(*args, **kwargs)
self.fields['my_field'].choices = my_field_choices
您的formset需要覆盖一个方法来运行查询集,并在构造它们时将它们传递给表单:
from django.forms.models import BaseModelFormSet
class MyFormSet(BaseModelFormSet):
def _construct_forms(self):
# instantiate all the forms and put them in self.forms
self.forms = []
# Define each of your choices querysets
my_field_choices = Model.object.filter(...)
#Add your querysets to a dict to pass to the form
form_defaults = {'my_field_choices': my_field_choices, }
for i in xrange(min(self.total_form_count(), self.absolute_max)):
self.forms.append(self._construct_form(i, **form_defaults))
(请参阅the Django source了解这将如何运作)