ValueError:使用makemigrations时无法序列化函数:lambda

时间:2018-04-04 04:46:39

标签: python django python-3.x django-models

当我python manage.py makemigrations时,我遇到上述错误,我不确定错误发生在哪里。我看到一些关于这个问题的帖子,但我主要在DateTimeField()找到函数传递但在我的情况下我使用了auto_now属性而不是一些与datetime相关的函数。

但是,我在类方法中使用了lambda函数,如下所示。

@classmethod
    def get_content_models(cls):
        """
        Return all Package subclasses.
        """
        is_content_model = lambda m: m is not Package and issubclass(m, Package)


def set_helpers(self, context):
    current_package_id = getattr(current_package, "id", None)
    current_parent_id = getattr(current_package, "parent_id", None)
    self.is_current_child = self.parent_id == current_package_id
    self.is_child = self.is_current_child

    def is_c_or_a(package_id):
        parent_id = context.get("_parent_package_ids", {}).get(package_id)
        return self.id == package_id or (parent_id and is_c_or_a(parent_id))
    self.is_current_or_ascendant = lambda: bool(is_c_or_a(current_package_id))

我不清楚这个问题,所以我已经发布了解原因。以上代码是否创建了该问题?如果是原因,应该做什么呢?

我不知道这个问题究竟在哪里,所以这里是gist中models.py of package app的详细信息,因为代码有点大。它没有达到预订模型,所以我只是把包的应用程序的代码,这里是

https://gist.github.com/SanskarSans/51d2f287309a97163e680cc38abd3e06

更新

在我的Package模型中,我使用了自定义字段,在该字段中使用了lambda而不是callable函数,这就产生了一个问题。由于模型的长文件,我没有在这里粘贴,我为此道歉。

这就是我做过的事情

in_menus = MenusField(_("Show in menus"), blank=True, null=True)

class MenusField(MultiChoiceField):
    """
    ``MultiChoiceField`` for specifying which menus a package should
    appear in.
    """

    def __init__(self, *args, **kwargs):
        choices = [t[:2] for t in getattr(settings, "PAGE_MENU_TEMPLATES", [])]
        default = getattr(settings, "PAGE_MENU_TEMPLATES_DEFAULT", None)
        if default is None:
            default = [t[0] for t in choices]
        elif not default:
            default = None
        if isinstance(default, (tuple, list)):
            d = tuple(default)
            # this lambda should be avoided
            # default = lambda:d 
            default = default_value(d)
        defaults = {"max_length": 100, "choices": choices, "default": default}
        defaults.update(kwargs)
        super(MenusField, self).__init__(*args, **defaults)

3 个答案:

答案 0 :(得分:0)

我假设您正在使用pickle模块进行序列化。

您无法挑选定义set_helpers的班级。该方法将self.is_current_or_ascendant设置为lambda函数,并且这些函数列在无法进行pickle的事物列表中(参见https://docs.python.org/3/library/pickle.html中的12.1.4)。

类方法不能成为问题,因为它只定义了一个局部变量is_content_model,它立即超出范围并被删除。事实上,正如你在这里提出的那样,这个方法什么都不做。

答案 1 :(得分:0)

lambda替换为函数的示例。

破解版本:

class SomeModel(ParentModel):
    thing_to_export = ArrayField(models.CharField(max_length=50), 
                                 default=lambda: ['Default thing'])

工作版本:

def default_thing():
    return ['THIS IS A DEFAULT']

class SomeModel(ParentModel):
    thing_to_export = ArrayField(models.CharField(max_length=50), 
                                 default=default_thing)

答案 2 :(得分:0)

以下内容可能对将来的其他人有所帮助。

它可以将lambda表达式更改为function

以下square_rootsquare_root_lambda的工作方式相同:

def square_root(x):
    return math.sqrt(x)

square_root_lambda = lambda x: math.sqrt(x)


print(square_root(4))
print(square_root_lambda(4))