访问全局变量时模块的 __setattr__ 和 __getattr__
Posted
技术标签:
【中文标题】访问全局变量时模块的 __setattr__ 和 __getattr__【英文标题】:A module's __setattr__ and __getattr__ when accessing globals 【发布时间】:2011-05-24 20:38:45 【问题描述】:__getattr__
和__setattr__
在我使用点符号获取或设置对象的属性时效果很好。
如何从模块内部拦截获取和设置模块全局变量?
I 子类化模块类型:
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()
中不存在的一些变量,我想替换请求的值。
更新:
上面的代码不能按我的需要工作:print
没有反映对变量 a
的访问。
我正在使用 PyQt4 编写一个“平台”,其他用户(程序员)在其中添加处理与这些表单交互的表单和模块。表单本身可通过“注入”变量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)
...
【问题讨论】:
您的实际问题是什么,因为您似乎已经回答了“我如何从模块内部拦截获取和设置模块全局变量?”? 【参考方案1】:你不能。
基本问题是属性重载仅在您访问属性时才起作用,特别是这意味着:
expr . ident没有点,就不会有属性重载。因此,无论你做什么,一个序列像
一个永远不能调用属性重载,无论其他什么可能等同于a
的值。
【讨论】:
【参考方案2】:在一般情况下,您尝试执行的操作无法完成。你不能让一个属性既有值又有属性。您可以同时使用form.myCheckbox = True
和bool(form.myCheckbox)
,或 form.myCheckbox.setChecked(True)
和bool(form.myCheckbox.isChecked())
,不是。我的意思是,form.myCheckbox
不能同时是 True
和 QCheckbox
对象。你可以让bool
、int
和unicode
工作,可以处理但不方便,而其他类型通常是不可能的。
除此之外,您对__getattr__
的作用感到困惑。存在的属性不会调用它,__getattribute__
是。但是使用__getattribute__
很复杂,您可能希望使您的复选框成为描述符或使用属性,但是您必须放弃模块并使用类。
附:为什么是模块,而不是某个对象的实例?
【讨论】:
form
是从 .ui 文件加载的 QDialog。我使用__getattr__
来拦截尚未定义的属性以上是关于访问全局变量时模块的 __setattr__ 和 __getattr__的主要内容,如果未能解决你的问题,请参考以下文章
Python魔法方法之属性访问 ( __getattr__, __getattribute__, __setattr__, __delattr__ )