来自Decorator的Python导入模块

时间:2015-03-12 19:34:09

标签: python python-3.x python-import python-decorators

我正在使用Python 3中的应用程序,我正在做的事情是非常规的。

cx_Oracle是一个难以设置的模块,对我的应用程序来说是一个可选的依赖项。我想要做的是将模块的导入包装在装饰器中,仅放置在使用它的函数之上。这将不必在我的模块顶部导入,并允许它不被设置。

class Loader():
    def load_cx_oracle(func):
        def inner(*args, **kwargs):

            # Additional code before import.

            import cx_Oracle

            return func(*args, **kwargs)
        return inner

    @load_cx_oracle
    def function_using_cx_oracle(self):
        conn = cx_Oracle.connect()

然而,当我尝试上述操作时,我得到NameError: name 'cx_Oracle' is not defined

3 个答案:

答案 0 :(得分:4)

接受的答案有几个问题。其中最重要的是每次调用函数时都会运行导入逻辑。第二个是装饰器必须在它使用的相同模块中定义,否则装饰器和装饰将具有不同的全局变量。您可以通过函数的__globals__属性直接访问函数的全局变量。代码示例首先在执行导入逻辑之前检查函数的全局变量中是否存在模块。该示例还使用functools.wraps装饰器在使用help(func)之类的内容时保留doc字符串,函数名称和参数名称。

from functools import wraps

def load_operator(func):
    @wraps(func)
    def wrapper(*args, **kwargs):
        if "operator" not in func.__globals__:
            # setup logic -- only executed once
            import operator
            func.__globals__["operator"] = operator
        return func(*args, **kwargs)
    return wrapper

class A:
    @load_operator
    def add(self, x, y):
        return operator.add(x, y)

    def subtract(self, x, y):
        return operator.subtract(x, y)

a = A()
try:
    a.subtract(1, 2)
except NameError:
    print("operator not yet loaded")
print(a.add(1, 2))
print(A.add)

答案 1 :(得分:3)

import仅绑定您使用它的命名空间中的模块名称。如果你在函数中执行import cx_Oracle,则名称cx_Oracle将仅在该函数内可用。

但是,您可以使用global进行全局分配。更改装饰器使用:

global cx_Oracle
import cx_Oracle

这种方法是否真的是正确的方法是值得商榷的。根据您的代码的使用方式,如果用户希望cx_Oracle可用,则可以更清楚地使用一个函数,如Daniel的答案所示,使用try-wrapped导入,或者通过某些外部设置(例如,配置文件)来定义加载。

答案 2 :(得分:0)

如果您想要可选的导入,请在模块的开头使用try-except:

try:
    import cx_Oracle
except ImportError:
    cx_Oracle = None