将 class 名称存储在 class 变量中而不键入 class 的名称?
Storing the class name in class variable without typing the name of the class?
在 python class 的实例方法中,我知道我们可以通过 self.__class__.__name__
获取 class 名称
但是,我想将 class 的名称存储在 class 变量中,而不对 class 的名称进行编码。
我知道我可以这样做来将 class 的名字放入 class 变量中:
class MyClass(object):
pass
MyClass._myname = MyClass.__name__
但是,我必须在 class 定义之外对字符串 MyClass
进行两次编码,只是为了将 class 名称放入 class 变量中。在这种情况下,这将是 "MyClass"
,我可以简单地编写如下代码:
class MyClass(object):
_myname = "MyClass"
但即使这样也是多余的,因为我必须在 MyClass
class.
中硬编码 "MyClass"
我想以某种方式将 class 的名称放入 class 变量中,而无需对 class 的名称进行编码,如下所示:
class MyClass(object):
_myname = ???? # where ???? is a statement which returns the class
# name, in which the string "MyClass" does not appear
这可能吗?
你可以使用装饰器
>>> def autoname(cls):
... cls._MyName=cls.__name__
... return cls
...
>>> @autoname
... class check:
... pass
...
>>> check._MyName
'check'
>>>
或者如果您不想对属性名称进行硬编码:
>>> def autoname2(name):
... def autoname(cls):
... setattr(cls, name, cls.__name__)
... return cls
... return autoname
...
>>> @autoname2('_StoreHere')
... class check2:
... pass
...
>>> check2._StoreHere
'check2'
>>>
当然第二种形式可以带更多的参数,例如:
>>> def autoname3(name, f=lambda x: x):
... def autoname(cls):
... setattr(cls, name, f(cls.__name__))
... return cls
... return autoname
...
>>> @autoname3('_MyNameIs', lambda x: x.upper() + '_The_Bold')
... class check3:
... pass
...
>>> check3._MyNameIs
'CHECK3_The_Bold'
>>>
关于装饰器与元类的说明:元类非常酷,但是,如果不是严格需要它们,最好避免使用它们,因为臭名昭著的 metaclass conflicts.
您可以使用元class:
class NameMeta(type):
def __new__(meta, name, bases, dct):
dct['_name'] = name
return super().__new__(meta, name, bases, dct)
class Foo(metaclass=NameMeta):
pass
print(Foo._name) # prints "Foo"
请注意,将名称设置为 class 上的新属性有点傻,因为 __name__
已经存在。可能您应该更改正在编写的查看 _name
属性的任何代码,并让它查找 __name__
。
在 python class 的实例方法中,我知道我们可以通过 self.__class__.__name__
但是,我想将 class 的名称存储在 class 变量中,而不对 class 的名称进行编码。
我知道我可以这样做来将 class 的名字放入 class 变量中:
class MyClass(object):
pass
MyClass._myname = MyClass.__name__
但是,我必须在 class 定义之外对字符串 MyClass
进行两次编码,只是为了将 class 名称放入 class 变量中。在这种情况下,这将是 "MyClass"
,我可以简单地编写如下代码:
class MyClass(object):
_myname = "MyClass"
但即使这样也是多余的,因为我必须在 MyClass
class.
"MyClass"
我想以某种方式将 class 的名称放入 class 变量中,而无需对 class 的名称进行编码,如下所示:
class MyClass(object):
_myname = ???? # where ???? is a statement which returns the class
# name, in which the string "MyClass" does not appear
这可能吗?
你可以使用装饰器
>>> def autoname(cls):
... cls._MyName=cls.__name__
... return cls
...
>>> @autoname
... class check:
... pass
...
>>> check._MyName
'check'
>>>
或者如果您不想对属性名称进行硬编码:
>>> def autoname2(name):
... def autoname(cls):
... setattr(cls, name, cls.__name__)
... return cls
... return autoname
...
>>> @autoname2('_StoreHere')
... class check2:
... pass
...
>>> check2._StoreHere
'check2'
>>>
当然第二种形式可以带更多的参数,例如:
>>> def autoname3(name, f=lambda x: x):
... def autoname(cls):
... setattr(cls, name, f(cls.__name__))
... return cls
... return autoname
...
>>> @autoname3('_MyNameIs', lambda x: x.upper() + '_The_Bold')
... class check3:
... pass
...
>>> check3._MyNameIs
'CHECK3_The_Bold'
>>>
关于装饰器与元类的说明:元类非常酷,但是,如果不是严格需要它们,最好避免使用它们,因为臭名昭著的 metaclass conflicts.
您可以使用元class:
class NameMeta(type):
def __new__(meta, name, bases, dct):
dct['_name'] = name
return super().__new__(meta, name, bases, dct)
class Foo(metaclass=NameMeta):
pass
print(Foo._name) # prints "Foo"
请注意,将名称设置为 class 上的新属性有点傻,因为 __name__
已经存在。可能您应该更改正在编写的查看 _name
属性的任何代码,并让它查找 __name__
。