1

我有以下代码:

import threading
from functools import wraps
class Synchronized(object):
    def __init__(self):
        self.lock = threading.Lock()

    def synchronized(f):
        @wraps(f)
        def wrapper(self, *args, **kwargs):
            with self.lock:
                print "here"
                return f(self, *args, **kwargs)
        return wrapper

    @synchronized
    def go(self):
        print 1

class B(Synchronized):
    @synchronized
    def foo(self):
        return 1

此代码在导入抱怨时失败:

  File "a.py", line XXX, in B
    @synchronized
NameError: name 'synchronized' is not defined

但是,如果我注释掉 B 并使用Syncrhonized().go()它,效果会很好。

问题: python 如何知道@synchronized基类中有什么但无法在其派生类中解析它?

4

1 回答 1

4

synchronized被定义为Synchronized only的类体中的一个函数。

类体像定义类的函数一样执行;生成的本地命名空间用于形成类属性。这就是为什么在用作装饰器时synchronized仍然是本地名称的原因。您可以将其与在函数内部定义装饰器进行比较,然后尝试在该函数外部应用它;它不起作用,因为装饰器是一个局部变量。Synchronizedgo

您可以@Syncronized.synchronized.im_func在类中使用B.im_func从方法包装器中解开函数):

class B(Synchronized):
    @Synchronized.synchronized.im_func
    def foo(self):
        return 1

更好的是,根本不要在类内部定义装饰器,而是. Synchronized毕竟,这不是一种方法:

def synchronized(f):
    @wraps(f)
    def wrapper(self, *args, **kwargs):
        with self.lock:
            print "here"
            return f(self, *args, **kwargs)
    return wrapper

class Synchronized(object):
    def __init__(self):
        self.lock = threading.Lock()

    @synchronized
    def go(self):
        print 1

class B(Synchronized):
    @synchronized
    def foo(self):
        return 1
于 2013-07-23T13:15:13.353 回答