类方法和实例方法存在于同一个命名空间中,你不能重复使用这样的名称;在这种情况下,最后一次定义的id将会覆盖之前的定义。
类方法仍然可以在实例上正常工作,但是没有必要创建一个单独的实例方法;只需使用:
class X:
@classmethod
def id(cls):
return cls.__name__
由于该方法仍然与类绑定:
>>> class X:
... @classmethod
... def id(cls):
... return cls.__name__
...
>>> X.id()
'X'
>>> X().id()
'X'
这是明确记录的:
它可以在类上调用(例如:C.f()
)或在实例上调用(例如:C().f()
)。除了类别之外,实例被忽略。
如果您需要区分绑定到类和实例
如果您需要根据使用位置的不同而使方法以不同方式工作;当在类上访问时绑定到类,当在实例上访问时绑定到实例,则需要创建一个自定义的描述符对象。
描述符API 是Python将函数绑定为方法并将 classmethod
对象绑定到类的方式;请参阅 描述符指南。
您可以通过创建一个具有`__get__`方法的对象来为方法提供自己的描述符。以下是一个简单的示例,根据上下文将方法绑定到不同的对象。如果`__get__`的第一个参数为`None`,则描述符将被绑定到一个类;否则,它将被绑定到一个实例。
class class_or_instancemethod(classmethod):
def __get__(self, instance, type_):
descr_get = super().__get__ if instance is None else self.__func__.__get__
return descr_get(instance, type_)
这个方法重新使用了
classmethod
,只是重新定义了它处理绑定的方式,对于
instance is None
的情况,委托给原始实现,对于其他情况,则委托给标准函数
__get__
的实现。
需要注意的是,在方法本身中,你可能需要测试它绑定到了什么。
isinstance(firstargument, type)
是一个很好的测试方法:
>>> class X:
... @class_or_instancemethod
... def foo(self_or_cls):
... if isinstance(self_or_cls, type):
... return f"bound to the class, {self_or_cls}"
... else:
... return f"bound to the instance, {self_or_cls"
...
>>> X.foo()
"bound to the class, <class '__main__.X'>"
>>> X().foo()
'bound to the instance, <__main__.X object at 0x10ac7d580>'
另一种实现方式可以使用两个函数,一个用于绑定到类,另一个用于绑定到实例:
class hybridmethod:
def __init__(self, fclass, finstance=None, doc=None):
self.fclass = fclass
self.finstance = finstance
self.__doc__ = doc or fclass.__doc__
self.__isabstractmethod__ = bool(
getattr(fclass, '__isabstractmethod__', False)
)
def classmethod(self, fclass):
return type(self)(fclass, self.finstance, None)
def instancemethod(self, finstance):
return type(self)(self.fclass, finstance, self.__doc__)
def __get__(self, instance, cls):
if instance is None or self.finstance is None:
return self.fclass.__get__(cls, None)
return self.finstance.__get__(instance, cls)
这是一个带有可选实例方法的类方法。使用它就像使用一个 `property` 对象一样;用 `@.instancemethod` 装饰实例方法:
>>> class X:
... @hybridmethod
... def bar(cls):
... return f"bound to the class, {cls}"
... @bar.instancemethod
... def bar(self):
... return f"bound to the instance, {self}"
...
>>> X.bar()
"bound to the class, <class '__main__.X'>"
>>> X().bar()
'bound to the instance, <__main__.X object at 0x10a010f70>'
个人而言,我的建议是谨慎使用这个方法;根据上下文改变行为的完全相同方法可能会让人感到困惑。然而,这种情况确实存在用例,比如SQLAlchemy在模型中区分SQL对象和SQL值的方式,其中列对象会像这样切换行为;请参阅他们的《混合属性》文档(link1)。这个实现遵循了我上面介绍的
hybridmethod
类的完全相同模式。
以下是根据要求进行类型提示的版本。这些版本需要您的项目已安装
typing_extensions
:
from typing import Generic, Callable, TypeVar, overload
from typing_extensions import Concatenate, ParamSpec, Self
_T = TypeVar("_T")
_R_co = TypeVar("_R_co", covariant=True)
_R1_co = TypeVar("_R1_co", covariant=True)
_R2_co = TypeVar("_R2_co", covariant=True)
_P = ParamSpec("_P")
class class_or_instancemethod(classmethod[_T, _P, _R_co]):
def __get__(
self, instance: _T, type_: type[_T] | None = None
) -> Callable[_P, _R_co]:
descr_get = super().__get__ if instance is None else self.__func__.__get__
return descr_get(instance, type_)
class hybridmethod(Generic[_T, _P, _R1_co, _R2_co]):
fclass: Callable[Concatenate[type[_T], _P], _R1_co]
finstance: Callable[Concatenate[_T, _P], _R2_co] | None
__doc__: str | None
__isabstractmethod__: bool
def __init__(
self,
fclass: Callable[Concatenate[type[_T], _P], _R1_co],
finstance: Callable[Concatenate[_T, _P], _R2_co] | None = None,
doc: str | None = None,
):
self.fclass = fclass
self.finstance = finstance
self.__doc__ = doc or fclass.__doc__
self.__isabstractmethod__ = bool(getattr(fclass, "__isabstractmethod__", False))
def classmethod(self, fclass: Callable[Concatenate[type[_T], _P], _R1_co]) -> Self:
return type(self)(fclass, self.finstance, None)
def instancemethod(self, finstance: Callable[Concatenate[_T, _P], _R2_co]) -> Self:
return type(self)(self.fclass, finstance, self.__doc__)
@overload
def __get__(self, instance: None, cls: type[_T]) -> Callable[_P, _R1_co]: ...
@overload
def __get__(self, instance: _T, cls: type[_T] | None = ...) -> Callable[_P, _R1_co] | Callable[_P, _R2_co]: ...
def __get__(self, instance: _T, cls: type[_T] | None = None) -> Callable[_P, _R1_co] | Callable[_P, _R2_co]:
if instance is None or self.finstance is None:
return self.fclass.__get__(cls, None)
return self.finstance.__get__(instance, cls)