Python @classmethod

@classmethod:

定义操作类,而不是操作实例的方法。classmethod改变了调用方法的方式,因此类方法的第一个参数是类本身,而不是实例。@classmethod最常见的用途是定义备选构造方法。

1.定义方式

Python中3种方式定义类方法, 常规方式, @classmethod修饰方式, @staticmethod修饰方式
  • 普通的self类方法foo()需要通过self参数隐式的传递当前类对象的实例

  • @classmethod修饰的方法class_foo()需要通过cls参数传递当前类对象

  • @staticmethod修饰的方法定义与普通函数是一样的。

self和cls的区别不是强制的,只是PEP8中一种编程风格,self通常用作实例方法的第一参数,cls通常用作类方法的第一参数。即通常用self来传递当前类对象的实例,cls传递当前类对象。

2. @classmethod 特点

  • 声明一个类方法
  • 第一个参数必须是 cls, 它可以访问类的属性
  • 类的方法只能访问类的属性,不能访问实例属性
  • 类的方法可以通过 ClassName.MethodName() 访问也可以通过实例来访问。
  • 它可以返回一个类的实例

例子:

class Student:
    name = 'unknown' # class attribute
    def __init__(self):
        self.age = 20  # instance attribute

    @classmethod
    def tostring(cls):
        print('Student Class Attributes: name=',cls.name)

类方法访问实例属性时,会报错

class Student:
    name = 'unknown' # class attribute
    def __init__(self):
        self.age = 20  # instance attribute

    @classmethod
    def tostring(cls):
        print('Student Class Attributes: name=',cls.name,', age=', cls.age)
>>> Student.tostring()
Traceback (most recent call last):
  File "<pyshell#22>", line 1, in <module>
    Student.tostring()
  File "<pyshell#21>", line 7, in display
    print('Student Class Attributes: name=',cls.name,', age=', cls.age)
AttributeError: type object 'Student' has no attribute 'age'

例子

from datetime import date

# random Person
class Person:
    def __init__(self, name, age):
        self.name = name
        self.age = age

    @staticmethod
    def fromFathersAge(name, fatherAge, fatherPersonAgeDiff):
        return Person(name, date.today().year - fatherAge + fatherPersonAgeDiff)

    @classmethod
    def fromBirthYear(cls, name, birthYear):
        return cls(name, date.today().year - birthYear)

    def display(self):
        print(self.name + "'s age is: " + str(self.age))

class Man(Person):
    sex = 'Male'

man = Man.fromBirthYear('John', 1985)
print(isinstance(man, Man))

man1 = Man.fromFathersAge('John', 1965, 20)
print(isinstance(man1, Man))

3. @classmethod 和 @staticmethod

@classmethod @staticmethod
声明一个类的方法 声明一个静态方法
可以访问类的属性,不可以访问实例属性 既不可以访问类属性,也不可以访问实例属性
可以使用 ClassName.MethodName() 调用或者 object.MethodName().调用 可以使用 ClassName.MethodName() 调用后者object.MethodName().调用
可以声明一个类的工厂方法,返回一个实例 不可以返回一个实例

参考:

https://www.programiz.com/python-programming/methods/built-in/classmethod
https://www.tutorialsteacher.com/python/classmethod-decorator

posted @ 2021-07-20 23:33  sixinshuier  阅读(893)  评论(0编辑  收藏  举报