How can I import a module dynamically given its name as string?
Как я могу импортировать модуль, которому динамически присваивается его имя в виде строки?
Я пишу приложение на Python, которое принимает команду в качестве аргумента, например:
$ python myapp.py command1
Я хочу, чтобы приложение было расширяемым, то есть могло добавлять новые модули, реализующие новые команды, без необходимости изменять основной исходный код приложения. Дерево выглядит примерно так:
Функция импортирует модуль name, потенциально используя заданный globals и locals, чтобы определить, как интерпретировать имя в контексте пакета. В fromlist задаются имена объектов или подмодулей, которые должны быть импортированы из модуля, указанного в name.
Это работает просто отлично, мне просто интересно, возможно ли более идиоматичный способ выполнить то, что мы делаем с этим кодом.
Обратите внимание, что я специально не хочу использовать eggs или точки расширения. Это не проект с открытым исходным кодом, и я не ожидаю, что там будут "плагины". Суть в том, чтобы упростить основной код приложения и устранить необходимость изменять его каждый раз, когда добавляется новый командный модуль.
The recommended way for Python 2.7 and 3.1 and later is to use importlib module:
importlib.import_module(name, package=None)
Import a module. The name argument specifies what module to import in absolute or relative terms (e.g. either pkg.mod or ..mod). If the name is specified in relative terms, then the package argument must be set to the name of the package which is to act as the anchor for resolving the package name (e.g. import_module('..mod', 'pkg.subpkg') will import pkg.mod).
e.g.
my_module = importlib.import_module('os.path')
Ответ 3
Note: imp is deprecated since Python 3.4 in favor of importlib
As mentioned the imp module provides you loading functions:
I've used these before to perform something similar.
In my case I defined a specific class with defined methods that were required. Once I loaded the module I would check if the class was in the module, and then create an instance of that class, something like this:
# Verify contents of the module: print(dir(module))
From here, module will be a module object representing the pluginX module (the same thing that would be assigned to pluginX by doing import pluginX). Thus, to call e.g. a hello function (with no parameters) defined in pluginX, use module.hello().
To get the effect "importing" functionality from the module instead, store it in the in-memory cache of loaded modules, and then do the corresponding from import:
sys.modules[module_name] = module
from pluginX import hello hello()
Importing a package
To import a package instead, calling import_module is sufficient. Suppose there is a package folder pluginX in the current working directory; then just do