Python装饰器函数范围

时间:2013-01-05 19:22:46

标签: python multithreading decorator scoping

我似乎无法弄清楚下面的代码是不是将demovar暴露给装饰函数:

def exposebasevar(function):
    def decorator(*args, **kwargs):
        demovar = 'Where am I?' # => or MyThreadSafeObjectHandle()
        return function(*args, **kwargs)
    return decorator

@exposebasevar
def usesexposedvariable():
    print demovar # this line will give an error

usesexposedvariable()

背景:我有一个模块,每个函数都使用一个全局变量。 现在我想使线程安全,所以我想通过装饰器使这个var可用。然后在装饰器中我可以找出对象的哪个实例属于一个线程并将其传入。这样我就不需要为每个函数更改签名或内容。

有人知道如何设置吗?感谢。

3 个答案:

答案 0 :(得分:3)

这将是动态范围。被广泛认为是一个非常糟糕的想法,因此在包括Python在内的各种语言中根本不受支持。只是不可能,除非你想拉非常肮脏和脆弱的黑客。我实际上将此作为标准免责声明,不是因为我有任何这样的解决方案。我能提供的最接近“工作”的是重写字节码,呃。

相反,将其设为装饰函数的参数。或者摆脱全球批发,从长远来看可能更简单,更好。

def exposebasevar(function):
    def decorator(*args, **kwargs):
        demovar = 'Where am I?'
        return function(*args, demovar=demovar, **kwargs)  # <<
    return decorator

@exposebasevar
def usesexposedvariable(demovar):  # <<
    print demovar

答案 1 :(得分:2)

您的代码无法正常运行,原因如下:

def f():
  print v   # NameError: global name 'v' is not defined

def g():
  v = 42
  f()

g()

最干净的方法是将demovar作为参数传递给function

def exposebasevar(function):
    def decorator(*args, **kwargs):
        demovar = 'Where am I?' # => or MyThreadSafeObjectHandle()
        return function(*args, demovar=demovar, **kwargs)
    return decorator

@exposebasevar
def usesexposedvariable(demovar):
    print demovar # this line will give an error

usesexposedvariable()

答案 2 :(得分:1)

  

我有一个模块,其中包含一个全局var,用于每个函数。现在我想让线程安全......

你所描述的基本上是threading.local的目的;不幸的是,您可能不得不将代码更改为 little 以使其正常工作,问题在于线程本地对象本身不是线程本地的;只有它的属性。只要你总是使用线程局部对象的属性,它几乎是透明的: