我正在使用 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
已接受的答案存在一些问题。其中最重要的是,每次调用函数时它都会运行导入逻辑。第二个是装饰器必须定义在使用它的同一模块中,否则装饰器和被装饰器将具有不同的全局变量。您可以通过以下方式直接访问函数的全局变量__globals__
函数的属性。在执行导入逻辑之前,代码示例首先检查函数的全局变量中是否存在模块。该示例还使用了functools.wraps
装饰器在使用诸如以下内容时保留文档字符串、函数名称以及参数名称help(func)
.
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)
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)