__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 myCheckbox
或myCheckbox = myValue
,拦截值访问权限并在后台进行必要的工作:
def __getattr__(self, name):
formWidget = getatttr(form, name)
if isinstance(formWidget, QLineEdit):
formWidget.setText(value)
...
答案 0 :(得分:2)
你做不到。
基本问题是属性重载仅在您访问属性时有效,具体意味着:
expr . ident
没有点,就没有属性重载。因此,无论你做什么,像
这样的序列a
永远不能调用属性重载,无论其他什么可能等同于{em> a
的值。
答案 1 :(得分:0)
在一般情况下,您无法完成的工作。您不能让属性都具有值和属性。您可以执行form.myCheckbox = True
和bool(form.myCheckbox)
,或 form.myCheckbox.setChecked(True)
和bool(form.myCheckbox.isChecked())
,不。我的意思是,form.myCheckbox
不能同时是True
和QCheckbox
个对象。您可以使其适用于bool
,int
和unicode
可以处理但不方便,而其他类型通常是不可能的。
除此之外,你对__getattr__
的作用感到困惑。它不是为存在的属性调用的,__getattribute__
是。但是使用__getattribute__
很复杂,你可能想要制作复选框描述符或使用属性,但是你必须放弃模块并使用类。
P.S。为什么形成一个模块,而不是某个对象的实例?