使用 getattr
在模块中动态实现类似类的行为在某些情况下,可能需要模仿__getattr__ 对类的行为,但对整个模块的行为。这允许动态创建类实例并根据模块上的属性查找调用其方法。
但是,尝试直接在模块上定义 __getattr__ 方法面临两个障碍:
基于包装的解决方案
一种方法是围绕模块创建一个包装器。 Sys.modules 能够容忍不同的对象类型,因此我们可以将模块包装在一个类中并将其分配给 sys.modules[__name__]。这允许动态行为而无需修改模块本身。然而,这种技术仅适用于模块级访问。
Guido van Rossum 的 Hack
Guido van Rossum 建议的另一个解决方案涉及替换 sys.path 中的实际模块。具有模块内定义的类实例的模块。导入机制执行最后的替换步骤,从而实现此黑客攻击。以下示例演示了这种方法:
<code class="python"># module foo.py import sys class Foo: def funct1(self, *args): <code block> def funct2(self, *args): <code block> sys.modules[__name__] = Foo()</code>
现在,可以通过 Foo 的实例访问模块中定义的函数。
注意事项
使用这些技术时,在 sys.modules 分配后可能无法访问其他模块元素。在替换类中定义所有必要的功能,以避免丢失模块内容。
__all__ 属性
使用 from module import * 时,在替换类中定义 __all__ 来处理此问题导入声明的类型。省略 __all__ 中的 __module__ 和 __qualname__ 等属性。
以上是如何在 Python 模块中模仿类行为?的详细内容。更多信息请关注PHP中文网其他相关文章!