4

我只是想为一个小型个人 Web 应用程序试用小马 orm 映射器。除了为实体定义自定义构造函数外,一切正常。

在下面的代码中,我创建了一个带有字符串字段的简单实体name并定义了一个构造函数,它除了将参数重定向到父构造函数之外什么都不做(在我的真实应用程序中,我更改了一些参数,以便将它们传递给父构造函数)。之后我创建一个User并打印它的名字。

from pony.orm import Database, Required, db_session, commit


db = Database("sqlite", ":memory:", create_db=True)


class User(db.Entity):
    def __init__(self, *args, **kwargs):
        super(User, self).__init__(*args, **kwargs)

    name = Required(str)


db.generate_mapping(create_tables=True)

with db_session:
    u = User(name="Admin")
    commit()
    print(u, u.name)

错误消息TypeError: object.__init__() takes no parameters与调用在同一行super()。看起来关键字参数被发送到object而不是db.Entity. 当我删除构造函数时,一切正常

那么为什么它不起作用。像我的示例中的构造函数不应该总是工作(当然什么也不做)吗?ponyorm 中有什么东西阻止它工作还是我在这里遗漏了什么?

为了完整起见,我的实体定义实际上看起来像这样

class User(db.Entity):
    def __init__(self, *args, **kwargs):
        if "password" not in kwargs:
            raise ValueError("password is required")
        kwargs["password"] = werkzeug.security.generate_password_hash(kwargs["password"])
        super().__init__(*args, **kwargs)

    name = Required(str)
    password = Required(str)

它产生相同的结果。

同样在官方文档中,它说至少允许在实体中创建方法。http://doc.ponyorm.com/entities.html#adding-custom-methods-to-entities。但它没有说明构造函数。

4

1 回答 1

3

直到那一刻,Entity该类根本没有该__init__方法。由于历史原因,实体实例的初始化发生在__new__方法中。因此,当您调用时,super(User, self).__init__您实际上调用了该object.__init__方法。虽然在 Python 2 中该object.__init__方法静默接受任何参数,但在 Python 3 中,当您将任何参数传递给该方法时,该方法会引发错误(您遇到的错误)。

在查看您的问题后,我们更改了实体实例的初始化逻辑,现在我们__init__以传统方式使用方法。如果您从 GitHub 获取 PonyORM 的最新版本,您的代码应该可以正常工作。此更改将成为下一个版本 PonyORM 0.6.2 的一部分。

感谢您指出这一点!

于 2015-08-31T12:26:30.727 回答