python的@classmethod和@staticmethod的区别和使用

https://www.geeksforgeeks.org/class-method-vs-static-method-python/ 此链接中文章给出了简洁清晰的解释。在本文中,我将翻译其中的几个关键点,并通过代码解释究竟有什么不同。

翻译

class C(object):
    @classmethod
    def fun(cls, arg1, arg2, ...):
       ....
fun: function that needs to be converted into a class method
returns: a class method for function.
  • class method是和类绑定的方法,不是和类的对象(实例)绑定的方法
  • class method能够访问类的状态,因为它可以接受一个指向类的参数(cls),而不是指向类实例的参数(self)。
  • class method可以修改类的状态,并应用到所有的类实例上。
class C(object):
    @staticmethod
    def fun(arg1, arg2, ...):
        ...
returns: a static method for function fun.
  • class method也是和类绑定的方法,不是和类的对象(实例)绑定
  • class method不能访问类的状态
  • class method存在于类中是因为它是一个相关的函数

代码理解

class A(object):
    value = 42
    
    def m1(self):
        print(self.value)

    @classmethod
    def m2(cls):
        print(cls.value)
        cls.value += 10

    @staticmethod
    def m3(cls_instance):
        cls_instance.value -= 10

Part1

a = A() # 
a.m1 # <bound method A.m1 of <__main__.A object at 0x7fc8400b7da0>>
a.m1() # 42

m1()是类A中的普通方法,必须在实例化的对象上进行调用。如果使用直接A.m1()就会得到m1() missing 1 required positional argument: 'self'的错误信息。

Part2

A.m2 # <bound method A.m2 of <class '__main__.A'>>
A.m2() # 42
a.m1() # 52

m2()函数经过@classmethod修饰,可以直接被调用。而且由于m2()的参数为cls,它能够直接访问、修改A中通过代码定义的部分(value),并且这种修改会体现在所有相关的实例中。这就是为什么在调用了A.m2()之后a.m1()输出的数值发生改变。

此时如果执行以下代码:

b = A()
b.m1() # 52

我们可以观察到输出值也发生了改变。

Part3

A.m3 # <function __main__.A.m3>
A.m3(a) # process
a.m1() # 42

由于添加了@staticmethod进行修饰,m3()函数也能够直接调用。但是由于其参数设置不包含self/cls,因此无法访问到value的数值。但是m3()函数可以像普通函数一样直接调用,并对A的实例进行修改。

上述的函数也可以完全脱离类的定义:

def m3(A_instance):
    A_instance.value -= 10

但是这样在类外进行定义不如在类内定义更为清晰。作为类的静态方式,m3()所处理的数据一般是类相关的数据,因而定义在类内会更好。

  • 3
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
`@classmethod` 和 `@staticmethod` 是 Python 用来定义类方法和静态方法的装饰器。 `@classmethod` 装饰器用于定义类方法。类方法是绑定到类而不是实例的方法,可以通过类或实例调用。类方法的第一个参数通常被约定为 `cls`,表示类本身,而不是实例。类方法可以访问类的属性和调用其他类方法。 示例: ```python class MyClass: @classmethod def my_class_method(cls, arg1, arg2): # 类方法可以访问类的属性 print(cls.__name__) print(arg1, arg2) # 通过类调用类方法 MyClass.my_class_method('hello', 'world') # 输出: # MyClass # hello world # 也可以通过实例调用类方法 obj = MyClass() obj.my_class_method('hello', 'world') # 输出: # MyClass # hello world ``` `@staticmethod` 装饰器用于定义静态方法。静态方法不需要访问实例或类的状态,因此不需要传递 `self` 或 `cls` 参数。静态方法可以通过类或实例直接调用。 示例: ```python class MyClass: @staticmethod def my_static_method(arg1, arg2): print(arg1, arg2) # 通过类调用静态方法 MyClass.my_static_method('hello', 'world') # 输出: # hello world # 也可以通过实例调用静态方法 obj = MyClass() obj.my_static_method('hello', 'world') # 输出: # hello world ``` 总结: - `@classmethod` 用于定义类方法,第一个参数为类本身,可以访问类的属性和调用其他类方法。 - `@staticmethod` 用于定义静态方法,不需要访问实例或类的状态,可以通过类或实例直接调用。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值