【问题标题】:Monkey patching a Django form class?猴子修补 Django 表单类?
【发布时间】:2011-04-25 05:40:19
【问题描述】:

给定一个表单类(在你巨大的 Django 应用程序的深处)..

class ContactForm(forms.Form):
    name = ...
    surname = ...

考虑到您想在不扩展或修改表单类本身的情况下向该表单添加另一个字段,为什么以下方法不起作用?

ContactForm.another_field = forms.CharField(...)

(我的第一个猜测是 Django 使用的元类hackery 仅在第一次构造表单类时应用。如果是这样,是否有办法重新声明该类来克服这个问题?)

【问题讨论】:

  • 你几乎肯定是对的。这正是您无法轻松地将新字段添加到 models.Model 子类的原因。
  • 对于模型,即使猴子补丁有效,也会出现“syncdb”问题。但是在某些时候,猴子补丁可能会挽救生命,恕我直言。

标签: python django django-forms monkeypatching


【解决方案1】:

一些相关的定义出现在django/forms/forms.py 中。它们是:

  1. class BaseForm
  2. class Form
  3. class DeclarativeFieldsMetaclass
  4. def get_declared_fields

get_declared_fieldsDeclarativeFieldsMetaclass 调用并构造一个列表,其中包含按其创建计数器排序的字段实例。然后它将基类中的字段添加到此列表中,并将结果作为OrderedDict 实例返回,其中字段名称作为键。 DeclarativeFieldsMetaclass 然后将此值粘贴到属性base_fields 中并调用type 来构造类。然后它将类传递给widgets.py 中的media_property 函数,并将返回值附加到新类的media 属性。

media_property 返回一个属性方法,该方法在每次访问时重构媒体声明。我的感觉是它在这里不相关,但我可能是错的。

无论如何,如果您没有声明 Media 属性(并且没有任何基类这样做),那么它只会返回一个新的 Media 实例,而构造函数没有任何参数,我认为猴子修补一个新字段on 应该像手动将字段插入base_fields 一样简单。

ContactForm.another_field = forms.CharField(...)
ContactForm.base_fields['another_field'] = ContactForm.another_field

然后每个表单实例在BaseForm__init__ 方法中获得base_fields 中的deepcopy 成为form_instance.fields。 HTH。

【讨论】:

  • +1 指向 OrderedDict 的点将我引向 SortedDict。不确定有什么区别,但它们都适用于我的问题(不了解 OP)。谢谢。
猜你喜欢
  • 2016-10-30
  • 1970-01-01
  • 2023-03-31
  • 2014-03-05
  • 2012-10-13
  • 2012-03-13
  • 2013-12-26
  • 2012-06-14
  • 2012-03-29
相关资源
最近更新 更多