将类的名称传递给函数,并在方法外部分配给类属性

时间:2019-04-16 17:02:38

标签: python class python-3.7

Python 3.7:

我有一个类定义,其中基于类名称为类属性分配了一个值,并且该值在类内的装饰器中使用。

我正在寻找“当前正在定义的班级”的代名词。

这是一个例子:

class MyClass:

    # NameError: name '__class__' is not defined
    class_logger = find_logger(__class__.__name__)

    # __init__ is not important
    def __init__(self):
        # This works!
        self.logger = find_logger(__class__.__name__)

    @mydecorator(class_logger)
    # mymethod is not important
    def mymethod(self):
        return 1

“当前正在定义的类”的合适代词是什么?

(这类似于How to get the current Python class name in __init__ regardless of the class of "self"?,但在那里,问题是“在 init 的定义中”,当时是2.7。Python3.x解决了该问题)

2 个答案:

答案 0 :(得分:4)

当类定义尚未完成评估时,不能保证类对象存在并且用户可以访问。即使可以访问它,也可能无法对其进行初始化,以至于您无法对其做任何有用的事情。

也就是说,您也许可以获取类的名称。 __qualname__是一个属性,用于保存类/函数/方法/描述符/生成器实例的qualified name。您应该可以从类的定义中访问它。

class Fred:
    print("Fred's qualname:", __qualname__)
    class Barney:
        print("Barney's qualname:", __qualname__)

结果:

Fred's qualname: Fred
Barney's qualname: Fred.Barney

如果您的类不是在文件级别定义的,则可能需要执行一些字符串操作以将其名称与路径的其余部分分开。例如,将上面的示例更改为print("Barney's qualname:", __qualname__.rpartition(".")[-1]),仅获得“ Barney”而不是“ Fred.Barney”。


尽管我尽了最大的努力,但找不到能明确确认__qualname__作为常规名称而非属性访问的值的文档。我不完全相信这是定义明确的行为,因此我认为我不能毫无保留地将其用于生产质量代码。

答案 1 :(得分:2)

要在定义类名称时使用它,还可以使用带有__prepare__方法的元类:

def find_logger(name):
    return name + '_logger'


class PrepareClassLogger(type):
    def __prepare__(name, bases, **kwargs):
        return {'class_logger': find_logger(name)}


class MyClass(metaclass=PrepareClassLogger):
    pass

print(MyClass.class_logger)  # MyClass_logger