15

鉴于我有一个模块的代码对象,我如何获得相应的模块对象?

看起来moduleNames = {}; exec code in moduleNames做的事情非常接近我想要的。它将模块中声明的全局变量返回到字典中。但是如果我想要实际的模块对象,我该如何得到它呢?

编辑:看起来你可以滚动你自己的模块对象。模块类型没有方便地记录,但您可以执行以下操作:

import sys
module = sys.__class__
del sys
foo = module('foo', 'Doc string')
foo.__file__ = 'foo.pyc'
exec code in foo.__dict__
4

1 回答 1

25

正如评论已经指出的那样,在今天的 Python 中,实例化没有内置名称的类型的首选方法是调用通过标准库中的types模块获得的类型:

>>> import types
>>> m = types.ModuleType('m', 'The m module')

请注意,这不会自动将新模块插入sys.modules

>>> import sys
>>> sys.modules['m']
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
KeyError: 'm'

这是您必须手动执行的任务:

>>> sys.modules['m'] = m
>>> sys.modules['m']
<module 'm' (built-in)>

这可能很重要,因为模块的代码对象通常在模块添加到之后sys.modules执行——例如,这样的代码引用 是完全正确的,如果你忘记了这一步 sys.modules[__name__],那将失败 ( )。此步骤之后,并按照您在编辑中已有的设置,KeyErrorm.__file__

>>> code = compile("a=23", "m.py", "exec")
>>> exec code in m.__dict__
>>> m.a
23

(或 Python 3 的等价函数 where execis a function,如果 Python 3 是您使用的,当然;-) 是正确的(当然,您通常会通过比编译字符串更微妙的方式获得代码对象,但是这对您的问题不重要;-)。

在旧版本的 Python 中,您会在开始时使用new模块而不是types模块来创建新的模块对象,但new自 Python 2.6 起已弃用并在 Python 3 中删除。

于 2010-02-23T00:34:56.870 回答