阻止Django查询ModelFormSet中每个表单的ForeignKey选项

时间:2013-03-04 13:56:26

标签: django django-forms django-database

我正在为我的Django应用程序构建一个csv导入表单,并希望在ModelFormSet中显示要导入的行以用于验证目的。

因此,我向相关ModelAdmin添加了一个视图,该视图从csv读取行并打印ModelFormSet(queryset=an_empty_queryset, initial={data_from_the_csv})

问题是模型通过ForeignKey字段引用了其他三个模型,并且对于表单集中每个表单中的每个字段,都会发出数据库查询以填充{{1选项。

为什么Django不缓存表单(因为它被多次使用)或者是否已经有办法实现这一点我还不知道呢?

1 个答案:

答案 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了解这将如何运作)