从模块导入其名称的子模块

时间:2016-01-22 05:40:58

标签: python python-import

我在子模块prepared_db中有一个名为db.db_1的函数:

from spam import db

submodule_name = "db_1"
func_name = "prepare_db"
func = ...

如何通过上面的上下文中的子模块名称和函数名来获取函数?

更新

要回复@histrio的回答,我可以验证他的代码适用于os模块。但它在这种情况下不起作用。要创建示例:

$ mkdir -p spam/db
$ cat > spam/db/db_1.py
def prepare_db():
    print('prepare_db func')
$ touch spam/db/__init__.py
$ PYTHONPATH=$PYTHONPATH:spam

现在,您可以正常导入:

>>> from spam.db.db_1 import prepare_db
>>> prepare_db()
prepare_db func

但是如果你动态地这样做,我会收到这个错误:

>>> getattr(getattr(db, submodule_name), func_name)
---------------------------------------------------------------------------
AttributeError                            Traceback (most recent call last)
<ipython-input-9-1b6aa1216551> in <module>()
----> 1 getattr(getattr(db, submodule_name), func_name)

AttributeError: module 'spam.db.db_1' has no attribute 'prepared_db'

2 个答案:

答案 0 :(得分:3)

这很简单。您可以将模块视为对象。

import os

submodule_name = "path"
func_name = "exists"

submodule = getattr(os, submodule_name)
function = getattr(submodule, func_name)
function('/home') # True

或[只是为了好玩,不要那样做]

fn = reduce(getattr, ('sub1', 'sub2', 'sub3', 'fn'), module)

<强>更新

import importlib
submodule = importlib.import_module('.'+submodule_name, module.__name__)
function = getattr(submodule, func_name)

答案 1 :(得分:1)

我想我想通了,你需要将函数添加到__all__文件中的__init__.py变量,这样就可以了:

from .db_1 import prepare_db

__all__ = ['prepare_db']

之后,它应该可以正常工作。