使用 getattr
在模块中动态实现类类行为 在某些情况下,可能需要模仿__getattr__ 对类的行为,但对整个模块的行为。这允许动态创建类实例并根据模块上的属性查找调用其方法。
但是,尝试直接在模块上定义 __getattr__ 方法面临两个障碍:
基于包装器的解决方案
一种方法是围绕模块创建包装器。 Sys.modules 能够容忍不同的对象类型,因此我们可以将模块包装在一个类中并将其分配给 sys.modules[__name__]。这允许动态行为而无需修改模块本身。然而,这种技术仅适用于模块级访问。
Guido van Rossum's Hack
Guido van Rossum 建议的另一个解决方案涉及替换 sys.path 中的实际模块。具有模块内定义的类实例的模块。导入机制执行最后的替换步骤,从而实现此黑客攻击。以下示例演示了这种方法:
# module foo.py
import sys
class Foo:
def funct1(self, *args):
def funct2(self, *args):
sys.modules[__name__] = Foo()
现在,可以通过 Foo 的实例访问模块中定义的函数。
注意事项
使用这些技术时,在 sys.modules 分配后可能无法访问其他模块元素。在替换类中定义所有必需的功能,以避免丢失模块内容。
__all__ 属性
使用 from module import * 时,在替换类中定义 __all__ 来处理此问题导入声明的类型。省略 __all__ 中的 __module__ 和 __qualname__ 等属性。
免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。
Copyright© 2022 湘ICP备2022001581号-3