点击上方“AirPython”,选择“加为星标”
第一时间关注 Python 技术干货!
前几天,有一个小伙伴过来问我,Python 中的 @staticmethod、@classmethod、self、cls 分别代表什么意思,自己平时光顾着用,不知道具体的含义?
事实上,由于 Python 语言的灵活性,这部分内容在日常编码过程中,很容易被忽略掉
本篇文章将和大家一起聊聊这几个小知识点
装饰器 @staticmethod 修饰的方法称为:静态方法,和普通的函数没有什么区别
下面将聊聊实际项目中几种应用场景
1、要调用一个静态方法,一般使用形式是:「 类名.方法名() 」
class Web(object): @staticmethod def foo_staticmethod(): """静态方法""" pass if __name__ == '__main__': # 直接使用类名+方法名调用 Web.foo_staticmethod()当然,也可以实例化一个类对象,通过这个对象去调用静态方法,但是不建议使用这种方式
# 实例化一个对象 instance = Web() # 使用实例对象去调用静态方法(不建议) instance.foo_staticmethod()2、针对类中定义的静态变量,可以使用「 类名.变量名 」 的形式去访问
class Web(object): # 静态变量(类变量) name = "Python_Web" @staticmethod def foo_staticmethod(): """静态方法""" # 引用静态变量 print(Web.name)3、静态方法内部使用其他静态方法、类方法,同样是使用「 类名.方法名() 」去调用
class Web(object): # 静态变量(类变量) name = "Python_Web" # 类方法 @classmethod def foo_classmethod_other(cls): print('类方法被调用!') # 另外一个静态方法 @staticmethod def foo_staticmethod_other(): print('另外一个静态方法被调用!') @staticmethod def foo_staticmethod(): """静态方法""" # 调用其他静态方法 print(Web.foo_staticmethod_other()) # 调用类方法 print(Web.foo_classmethod_other())4、静态方法内部调用普通方法,访问实例属性
普通方法和实例属性都必须通过实例对象去引用,不能直接使用类名去访问
class Web(object): def __init__(self): self.desc = "实例属性,不共享" def norm_method(self): """普通方法""" print('普通方法被调用!') @staticmethod def foo_staticmethod(): """静态方法""" instance = Web() # 获取实例属性 print(instance.desc) # 调用普通方法 instance.norm_method()5、子类的使用
在子类中调用父类定义好的静态方法,只需要将类名替换为子类名称即可
class Web(object): @staticmethod def foo_staticmethod(arg1, arg2): pass class Django(Web): """子类""" pass if __name__ == '__main__': # 1、使用类名(父类)去调用静态方法 Web.foo_staticmethod("Hello", ",AirPython") # 2、使用类名(子类)去调用静态方法 Django.foo_staticmethod("Hello", ",AirPython")装饰器 @classmethod 修饰的方法称为:类方法,在使用的时候,会将类本身作为第一个参数 cls 传递给类方法
# 类方法,第一个参数为cls,代表类本身 @classmethod def foo_classmethod(cls): pass其中,cls 代表外层类本身,可以实例化,也可以直接调用静态方法、类方法、静态变量
下面逐一进行说明
1、要调用一个类方法,一般使用形式是:「 类名.方法名() 」
class Web(object): # 类方法,第一个参数为cls,代表类本身 @classmethod def foo_classmethod(cls): pass if __name__ == '__main__': # 使用类名去调用类方法 Web.foo_classmethod()和静态方法类似,也可以实例化一个类对象,通过这个对象去调用静态方法,但是不建议使用这种方式
2、调用静态变量
静态方法内部引用静态变量有两种方式,分别是:
「 类名.变量名 」
「 cls.变量名 」
注意:由于 cls 代表就是外层类本身,所以这两种方式等效
class Web(object): # 静态变量(类变量) name = "Python_Web" # 类方法,第一个参数为cls,代表类本身 @classmethod def foo_classmethod(cls): # 调用静态变量方式一 print(cls.name) # 调用静态变量方式二 print(Web.name)3、类方法内部调用其他类方法、静态方法
在一个类方法内部,可以使用「 类名.类方法名() 」、「 类名.静态方法名() 」的形式去调用方法
class Web(object): # 静态方法 @staticmethod def foo_staticmethod(): print('静态方法被调用!') # 其他类方法 @classmethod def foo_classmethod_other(cls): print('另外一个类方法被调用!') # 类方法,第一个参数为cls,代表类本身 @classmethod def foo_classmethod(cls): # 调用其他类方法 cls.foo_classmethod_other() # 调用静态方法 cls.foo_staticmethod() if __name__ == '__main__': Web.foo_classmethod()4、类方法内部调用普通方法,访问实例属性
需要通过 cls 变量实例化一个类对象,然后通过这个对象去调用普通方法和实例属性
class Web(object): def __init__(self): self.desc = "实例属性,不共享" def norm_method(self): """普通方法""" print('普通方法被调用!') # 类方法,第一个参数为cls,代表类本身 @classmethod def foo_classmethod(cls): # 如果要调用实例属性,必须使用cls实例化一个对象,然后再去引用 print(cls().desc) # 如果要调用普通方法,必须使用cls实例化一个对象,然后再去引用 cls().norm_method()5、子类的使用
在子类中调用父类定义好的类方法,只需要将类名替换为子类名称即可,代码和静态方法类似
下面总结一下普通方法、静态方法、类方法的区别
普通方法:第一个参数 self 代表实例对象本身,可以使用 self 直接引用定义的实例属性和普通方法;如果需要调用静态方法和类方法,通过「 类名.方法名() 」调用即可
静态方法:使用「 类名.静态变量 」引用静态变量,利用「 类名.方法名() 」调用其他静态方法和类方法;如果需要调用普通方法,需要先实例化一个对象,然后利用对象去调用普通方法
类方法:第一个参数 cls 代表类本身(等价),通过「 cls.静态变量 」或「 类名.静态变量 」引用静态变量,利用「 cls.方法名() 」或「 类名.方法名() 」去调用静态方法和类方法;如果需要调用普通方法,需要先实例化一个对象,然后利用对象去调用普通方法
静态方法和类方法是针对类定义的,除了可以使用类名去调用,也可以使用实例对象去调用,但是不建议
一般来说,如果方法内部涉及到实例对象属性的操作,建议用普通方法;如果方法内部没有操作实例属性的操作,仅仅包含一些工具性的操作,建议使用静态方法;而如果需要对类属性,即静态变量进行限制性操作,则建议使用类方法
我已经将文中全部源码上传到后台,关注公众号后回复「 pmethod 」即可获得全部源码
如果你觉得文章还不错,请大家 点赞、分享、留言 下,因为这将是我持续输出更多优质文章的最强动力!
推荐阅读
带你用 Python 实现自动化群控(入门篇)
Python 自动化,Appium 凭什么使用 UiAutomator2?
Python 自动化,Helium 凭什么取代 Selenium?