【问题标题】:Built-In callable as a default argument for a Django Field内置可调用作为 Django 字段的默认参数
【发布时间】:2020-08-21 19:11:06
【问题描述】:

我有一个 JSONField,我需要对其应用默认字典。根据文档,我避免将可变字典传递给默认字段。这是通过将复制方法传递给默认参数来完成的,如下所示:

default_dict = {'some_key': 'some value'}

class MyModel(models.Model):
    my_field = models.JSONField(default=default_dict.copy)

在应用 makemigrations 时,由于 django.db.migrations.serializer.FunctionTypeSerializer 中的以下条件而失败:

if self.value.__module__ is None:
    raise ValueError("Cannot serialize function %r: No module" % self.value)

我可以通过定义一个返回副本的可调用来解决这个问题,但我认为这会增加不必要的语法并使其更难阅读:

class ADict(dict):
    def __call__(self):
        return self.copy()

default_dict = ADict({'some_key': 'some value'})

class MyModel(models.Model):
    my_field = models.JSONField(default=default_dict)

有没有办法将内置对象方法作为 Django 字段的默认值传递?

【问题讨论】:

  • 你不能这样做,因为它基本上需要一个命名函数,而default_dict.copy是一个“匿名”函数。

标签: django django-models django-migrations


【解决方案1】:

你不能这样做,因为它基本上需要一个命名函数,而default_dict.copy是一个“匿名”函数。

但是,您可以创建一个命名函数,例如:

default_dict = {'some_key': 'some value'}

def copy_default_dict():
    return default_dict.copy()

class MyModel(models.Model):
    my_field = models.JSONField(default=copy_default_dict)

甚至更简单:

def copy_default_dict():
    return {'some_key': 'some value'}

class MyModel(models.Model):
    my_field = models.JSONField(default=copy_default_dict)

【讨论】:

  • 谢谢威廉,我就是这么想的。这种约束存在的原因是什么?看着“幕后”,我无法弄清楚为什么不允许这样做。删除异常意味着它正在尝试导入 None
  • @StuartBuckingham:这并不是真正的约束。问题是它需要出现在迁移文件中。使用命名函数,迁移会导入该函数。对于匿名函数,从 Django 的角度来看,它不可能重构事件序列来生成对象。
  • 如果是成员方法,比如mydict.copy,能不能只通过查看传入的匿名函数的self值然后getattr(mydict , func.__name__)?
  • @StuartBuckingham:匿名源于你有一个对象这一事实。请注意,Django 只是接收方法本身,而不是对象。所以你可以看到'copy' 方法,但是来自什么对象呢?如果它是{**foo, **bar}.copyfoobar 字典,那么该对象是一个“临时”对象。
  • 是的。谢谢你的详细信息。
猜你喜欢
  • 2018-10-20
  • 1970-01-01
  • 2020-02-15
  • 2023-03-03
  • 1970-01-01
  • 2010-09-19
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多