访问全局变量时,模块的__setattr__和__getattr__

时间:2010-12-13 18:50:01

标签: python module

当我使用点表示法获取或设置对象的属性时,

__getattr____setattr__效果很好。 如何拦截从模块内部获取和设置模块全局变量?

我是子类模块类型:

from types import ModuleType
class WUserModule(ModuleType):
    def __init__(self, name):
        super().__init__(name)

    def __setattr__(self, name, value):
        if name in self.__dict__:
             super().__setattr__(name, value)
             return
        print('Set name {}, value {}'.format(name, str(value)))

    def __getattr__(self, name):
        print('Get name {}'.format(name))

创建一个空模块并将代码加载到其中:

module = WUserModule('form_module')
with open('user_module.py', 'r', encoding='utf8') as f:
    exec(f.read(), module.__dict__)

user_module.py加载的代码:

a=1
print(a)

我需要以某种方式拦截对变量a的访问。在加载的代码中,期望访问模块globals()中不存在的一些变量,并且我想替换所请求的值。

更新:

上面的代码无法正常工作:a未反映对变量print的访问。

我正在使用PyQt4编写一个'平台',其他用户(程序员)添加表单和模块来处理与这些表单的交互。表单本身可以通过'inject'变量form在用户模块中访问。我想让用户可以从表单小部件和更简单的方式访问值。我想提供一个快捷方式:if form.myCheckbox.isChecked()form.myCheckbox.setChecked(myValue),而不是写if myCheckboxmyCheckbox = myValue,拦截值访问权限并在后台进行必要的工作:

def __getattr__(self, name):
    formWidget = getatttr(form, name)
    if isinstance(formWidget, QLineEdit):
        formWidget.setText(value)
...

2 个答案:

答案 0 :(得分:2)

你做不到。

基本问题是属性重载仅在您访问属性时有效,具体意味着:

expr . ident

没有点,就没有属性重载。因此,无论你做什么,像

这样的序列
a

永远不能调用属性重载,无论其他什么可能等同于{em> a的值。

答案 1 :(得分:0)

在一般情况下,您无法完成的工作。您不能让属性都具有值和属性。您可以执行form.myCheckbox = Truebool(form.myCheckbox) form.myCheckbox.setChecked(True)bool(form.myCheckbox.isChecked())。我的意思是,form.myCheckbox不能同时是TrueQCheckbox个对象。您可以使其适用于boolintunicode可以处理但不方便,而其他类型通常是不可能的。

除此之外,你对__getattr__的作用感到困惑。它不是为存在的属性调用的,__getattribute__是。但是使用__getattribute__很复杂,你可能想要制作复选框描述符或使用属性,但是你必须放弃模块并使用类。

P.S。为什么形成一个模块,而不是某个对象的实例?