- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我四处寻找这个问题的答案,但找不到任何东西。如果之前有人问过这个问题,我深表歉意。
我知道有 3-4 种方法可以从父类强制执行子类的给定方法(编辑元类的 __new__
方法,挂接到 builtins.__build_class__
,使用 __init_subclass__
或使用 abc.abstractmethod
)我通常最终使用 __init_subclass__
,主要是因为易于使用,而且与@abc.abstractmethod
,子类的约束是根据子类定义而不是类实例化来检查的。示例:
class Par():
def __init_subclass__(self, *args, **kwargs):
must_have = 'foo'
if must_have not in list(self.__dict__.keys()):
raise AttributeError(f"Must have {must_have}")
def __init__(self):
pass
class Chi(Par):
def __init__(self):
super().__init__()
这个示例代码显然会抛出错误,因为 Chi
没有 foo
方法。尽管如此,我还是发现了一个事实,即可以使用一个简单的类装饰器来绕过上游类的这个约束:
def add_hello_world(Cls):
class NewCls(object):
def __init__(self, *args, **kwargs):
self.instance = Cls(*args, **kwargs)
def hello_world(self):
print("hello world")
return NewCls
@add_hello_world
class Par:
def __init_subclass__(self, *args, **kwargs):
must_have = "foo"
if must_have not in list(self.__dict__.keys()):
raise AttributeError(f"Must have {must_have}")
def __init__(self):
pass
class Chi(Par):
def __init__(self):
super().__init__()
c = Chi()
c.hello_world()
上面的代码运行没有问题。现在,忽略我装饰的类是 Par
这一事实(当然,如果 Par
是库代码,我什至可能无法将其作为用户代码访问开发人员),我无法真正解释这种行为。对我来说很明显,可以使用装饰器向现有类添加方法或功能,但我从未见过不相关的装饰器(只是打印 hello world
,甚至不会混淆类创建)禁用类中已经存在的方法。
这是预期的 Python 行为吗?或者这是某种错误?老实说,据我所知,这可能会带来一些安全问题。
这是否只发生在 __init_subclass__
数据模型上?还是也给其他人?
最佳答案
记住,装饰器语法只是函数应用:
class Par:
def __init_subclass__(...):
...
Par = add_hello_world(Par)
最初绑定(bind)到 Par
的类定义了 __init_subclass__
; add_hello_world
中定义的new 类没有,就是后修饰名Par
所指的类,你是的类子类化。
顺便说一句,您仍然可以通过 __init__
访问原始类 Par
。
显式调用装饰器:
class Par:
def __init_subclass__(self, *args, **kwargs):
must_have = "foo"
if must_have not in list(self.__dict__.keys()):
raise AttributeError(f"Must have {must_have}")
def __init__(self):
pass
Foo = Par # Keep this for confirmation
Par = add_hello_world(Par)
我们可以确认闭包保留了对原始类的引用:
>>> Par.__init__.__closure__[0].cell_contents
<class '__main__.Par'>
>>> Par.__init__.__closure__[0].cell_contents is Par
False
>>> Par.__init__.__closure__[0].cell_contents is Foo
True
如果您确实尝试对其进行子类化,您将得到预期的错误:
>>> class Bar(Par.__init__.__closure__[0].cell_contents):
... pass
...
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "tmp.py", line 16, in __init_subclass__
raise AttributeError(f"Must have {must_have}")
AttributeError: Must have foo
关于python - 类装饰器禁用 __init_subclass__,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57811837/
我在修补自定义类时遇到问题'__init_subclass__ .我认为这与我将修补函数绑定(bind)到类的方式有关: def _patched_initsubclass(cls, **kwargs
我四处寻找这个问题的答案,但找不到任何东西。如果之前有人问过这个问题,我深表歉意。 我知道有 3-4 种方法可以从父类强制执行子类的给定方法(编辑元类的 __new__ 方法,挂接到 builtins
我终于升级了我的 python 版本,并且发现了添加的新功能。除其他外,我对新的 __init_subclass__ 摸不着头脑。方法。来自文档: This method is called when
对于在子类上强制执行某些不变量的元类或 __init_subclass__ 应该引发什么异常,是否有任何好的约定? 对于这个答案,我会接受来自 Python 内置行为、PEP、标准库或完善的第三方库的
我想创建一个设置注册表。我还希望能够将设置分组到宏类别中。 以下简化示例适用于单个注册表: class RegistryHolder: registry = {} def __init
我在一个项目中使用 init_subclass ,当我遇到内置方法时有点犹豫,当代码第一次在解释器中运行时启动——没有通过实例化直接引用包含类或它枚举的子类。 谁能告诉我这是怎么回事,并指出任何安全使
我正在尝试让类装饰器工作。装饰器将向其所应用的类添加一个 __init_subclass__ 方法。 但是,当该方法动态添加到类中时,第一个参数不会绑定(bind)到子类对象。为什么会发生这种情况?
使用 type() 创建动态类时,如何将值传递给 __init__subclass__() 中的 kwargs 参数 class Base(): def __init_subclass__(c
我正在尝试编写一个简单的插件系统。 a)为此,我使用 3.6 魔术函数 __init_subclass__ https://stackoverflow.com/a/41924331/1506569 找
我注意到我不能以我想要的方式将 __init_subclass__ 与 Django 模型类一起使用。看起来元类在父类的 __init_subclass__ 方法运行时还没有完成子类的创建。虽然我明白
我的问题灵感来自这个 question . 问题在于 3 级类模型 - 终止类(3 级)仅应存储在注册表中,但 2 级是干扰性的并且也已存储,因为它们是 1 级的子类。 我想通过使用元类摆脱 1 级类
我正在尝试创建一个元类,但是当我将它分配给另一个类时,我收到错误 TypeError: __init_subclass__() takes no keyword arguments但我没有实现任何 _
我有以下代码来比较基类的当前(空)所需功能的实现与其子类,子类必须以某种不同的方式实现它们,以便在运行时被认为是可接受的。如果不使用 metaclass=ABCMeta 并在这些基类方法上实现 @ab
这个问题在这里已经有了答案: Why does a classmethod's super need a second argument? (1 个回答) 3年前关闭。 我尝试从 __init_sub
我正在测试一些 pyqt 小部件及其方法,但当我创建一个继承 QLineEdit 类的子类以便在单击鼠标右键时创建上下文菜单时,我一直遇到问题。此类将主窗口作为父窗口。 运行此代码会产生以下输出: T
我正在尝试将 python 3.6 的新 __init_subclass__ 功能 ( PEP 487 ) 与 abc 模块一起使用。它似乎没有用。以下代码: from abc import ABCM
我是一名优秀的程序员,十分优秀!