避免模块名称损坏的干净方法?
问题描述
我正在使用Python的ctype来连接一个C库文件,运行ctypesgen来创建我可以导入的包装器模块(例如api.py)。碰巧的是,许多C函数和宏的名称都以双下划线开头,当在类定义中使用时,它们被Python视为"私有",并调用名称损坏。例如,这是可行的:
import api
api.__someGetter()
但这不是
import api
class MyClass(object):
i = api.__someGetter()
AttributeError: 'module' object has no attribute '_MyClass__someGetter'
此起作用:
i = api.__dict__['__someGetter']()
但对我来说,它似乎不是很优雅,也不容易阅读--代码中将充斥着对字典键的调用。
有没有更好的方法来禁用或解决模块的名称损坏?
(它是一个C库这一事实可能与答案无关,但希望它解释了为什么我不能简单地更改命名约定。)
解决方案
出于好奇,这里有一个实现此效果的方法。它将模块包装在一个具有custom attribute access methods的类中,这个类只丢弃了损坏名称的第一部分。这当然是一种黑客行为,并不是在所有情况下都能起作用--我不推荐认真使用它!
class Wrapper(object):
def __init__(self, module):
for name in module.__dict__:
self.__dict__[name] = module.__dict__[name]
#__getattr__ is called only if regular access failed
def __getattr__(self, name):
try:
return self.__dict__['__'+name.split('__', 1)[1]]
except (IndexError, KeyError):
raise AttributeError, name
def __setattr__(self, name, value):
try:
self.__dict__['__'+name.split('__', 1)[1]] = value
except IndexError: #no double underscores to split
self.__dict__[name] = value
用法:
import cwrapper
api = Wrapper(cwrapper)
class MyClass(object):
i = api.__some_Getter() #works
相关文章