12

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

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

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

为什么 Django 不缓存表单(因为它被多次使用)或者已经有一种我不知道的方法来完成这个?

4

1 回答 1

17

Django 表单集只是将表单创建的所有细节委托给表单对象本身,并且各个表单实例不知道其他表单实例,因此每个实例都必须查询自己的选择也就不足为奇了。

缓存也可能有意想不到的副作用——例如,表单的__init__功能可能依赖于initial它接收到的数据,从而使缓存的form对象不正确。

减少查询数量的最佳方法是检索一次选择查询集,然后将它们传递给它们的构造函数中的表单类。这将需要定义一个 customModelForm和一个 custom 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

并且您的表单集将需要覆盖一个方法来运行查询集并将它们传递到构造它们的表单中:

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))

(请参阅Django 源代码以了解其工作原理)

于 2013-03-06T15:30:55.687 回答