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解决了该问题)
答案 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