清除方法来绕过模块的名称修改?

时间:2012-07-27 02:45:57

标签: python ctypes

我正在使用Python的ctypes来连接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库可能与答案无关,但希望它能解释为什么我不能只改变命名约定。)

2 个答案:

答案 0 :(得分:2)

您可以在Python中重新导出名称,将双下划线转换为单个下划线。

import api
for name in api.__dict__:
    if name.startswith('__'):
        globals()[name[1:]] = api.__dict__[name]

这会将api中的所有双下划线更改为当前模块中的单个下划线。然后可以从另一个模块导入该模块。

注意: 不正确命名C中以两个下划线开头的函数。请参见n1526的第7.1.3节“保留标识符”。在C中使用以双下划线开头的名称是“未定义的行为”,这是“不要这样做”的技术术语。

答案 1 :(得分:0)

为了好奇,这是一种实现这种效果的方法。它将模块包装在一个具有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