6.4 类方法和对象方法

方法的本质其实就是函数,在模块中定义就是函数,在类中定义就是类的方法。方法按照归属分类一般有3种:对象方法、类方法、静态方法。

6.4.1 对象方法

class Person:
    cls_attr = '我是类属性'

    def __init__(self):
        self.obj_attr = '我是对象属性'

    def eat(self):
        print('我在吃饭,别烦我。')


zhangsan = Person()
zhangsan.eat()

我在吃饭,别烦我。

在上面的代码中:
我们在用def eat(self):定义了一个函数,而这个函数是在类中定义的,则称为这个类的方法。这个方法内部只有一条命令:print('我在吃饭,别烦我。')。
之前我们已经学会了如何定义一个函数,实际上类中的方法其实就是一个函数,只不过:
1、方法在类的内部定义
2、方法的第一个形参必须要有,通常为self,代表的就是类实例化后的那个对象本身。
3、方法的调用为:对象.方法(参数),在调用时,开发者不需要再手动传入一个实参给前面说的self形参,这个行为是在实例化后自动实现的。而函数的调用方法为:函数名(参数)。
zhangsan.eat():这句代码是执行了zhangsan对象中的eat方法,结果就是执行了定义时的语句:print('我在吃饭,别烦我。')

6.4.2 类方法

如果在定义对象方法前面添加@classmethod的装饰器后,该方法就从对象方法变成了类方法。如下:

class Person:

    def __init__(self):
        self.name = '张三'

    @classmethod
    def eat(cls):
        print('我在吃饭,别烦我。')


zhangsan = Person()
zhangsan.eat()
Person.eat()

我在吃饭,别烦我。
我在吃饭,别烦我。

类方法的第一个参数同对象方法一样,调用时不需要传入,通常形参写成cls。类方法可以通过类名或者实例化后的对象调用。

6.4.3 静态方法

静态方法则是在定义方法时添加了@staticmethod装饰器的方法。

class Person:

    def __init__(self):
        self.name = '张三'

    @staticmethod
    def eat():
        print('我在吃饭,别烦我。')


zhangsan = Person()
zhangsan.eat()

我在吃饭,别烦我。

定义静态方法时,不需要再传入selfcls等特殊的形参,而可以像其它函数一样使用参数定义。实际上,静态方法已经完全就是一个普通函数了。

6.4.4 私有方法

如果在类中定义了以__开头的方法名,表示该方法是一个私有方法,在类的外部是不可以直接访问的。当然,和私有属性一样,还是可以通过_类名方法名进行访问的。而以_开头的方法名也是一样,可以直接在外部访问,但是不建议使用。

class MyClass2:

    def _fun1(self):
        print('fun1')

    def __fun2(self):
        print('fun2')


mc = MyClass2()
mc._fun1()
mc._MyClass2__fun2()
mc.__fun2()

fun1
fun2
Traceback (most recent call last):
File "E:\BaiduNetdiskWorkspace\FrbPythonFiles\study\main.py", line 13, in
mc.__fun2()
AttributeError: 'MyClass2' object has no attribute '__fun2'

6.4.5 对象方法、类方法、静态方法的区别

参数区别:
对象方法:
第一个形参必须是指向对象本身,通常为self,调用时不需要传入对应实参。
类方法:
第一个形参必须是指向类本身,通常为cls,调用时不需要传入对应实参。
静态方法:
形参定义同普通函数一样。
调用属性区别:
对象方法:
可以通过self.的方式(和第一形参对应)调用对象属性,但是不能调用类属性。
类方法:
可以通过cls.的方式(和第一形参对应)调用类属性,但是不能调用对象属性。
静态方法:
不能调用类属性和对象属性。
调用区别:
对象方法:
对象可以直接调用;类也可以调用,前提是对象方法中没有调用对象属性。
类方法:
对象和类都可以调用。无论类方法中有没有调用类属性。
静态方法:
对象和类都可以调用。但是在调用时,静态方法中无法调用对象或类属性。

总结性代码如下,其中会报错的代码就注释掉了:

class Person:
    cls_attr = '我是类属性'

    def __init__(self):
        self.obj_attr = '我是对象属性'

    def eat(self):
        # print(cls_attr)
        print(self.obj_attr)

    @classmethod
    def sleep(cls):
        print(cls.cls_attr)
        # print(self.obj_attr)

    @staticmethod
    def run():
        # print(cls_attr)
        # print(self.obj_attr)
        print('我是静态方法')


zhangsan = Person()
zhangsan.eat()
zhangsan.sleep()
zhangsan.run()
# Person.eat()
Person.sleep()
Person.run()

我是对象属性
我是类属性
我是静态方法
我是类属性
我是静态方法

从自动化办公到智能化办公