1.通常来说,我们使用一个类的方法时,首先要实例化这个类,再用实例化的类来调用其方法
class Test(object): """docstring for Test""" def __init__(self, arg=None): super(Test, self).__init__() self.arg = arg def say_hi(self): print('hello wrold') def main(): test = Test() #//1. 首先实例化test类 test.say_hi() #//2. 再调用类的方法 if __name__ == '__main__': main()
输出
hello wrold
2.而使用@staticmethod或@classmethod,就可以不需要实例化,直接类名.方法名()来调用。
这有利于组织代码,把某些应该属于某个类的函数给放到那个类里去,同时有利于命名空间的整洁。
class Test(object): name = 'stephen' """docstring for Test""" def __init__(self, arg=None): super(Test, self).__init__() self.arg = arg def say_hi(self): print('[1]hello wrold') @staticmethod def say_bad(): print('[2]say bad') print('[3]Test.name:%s'%Test.name) #//不需要实例化 Test().say_hi() #//先要实例化,才能调用say_hi() @classmethod def say_good(cls): print('[4]say good') print('[5]cls.name:%s'%cls.name) #//不需要实例化 cls.say_bad() #//不需要实例化 cls().say_hi() #//先要实例化,才能调用say_hi() def main(): Test.say_bad() print("----------------------------------") Test.say_good() if __name__ == '__main__': main()
输出结果
[2]say bad [3]Test.name:stephen [1]hello wrold ---------------------------------- [4]say good [5]cls.name:stephen [2]say bad [3]Test.name:stephen [1]hello wrold [1]hello wrold
3.注意我们在定义这些方法时的区别
- 类的普通方法,第一个参数需要self参数表示自身。
- @staticmethod不需要表示自身对象的self和自身类的cls参数,就跟使用函数一样。
- @classmethod也不需要self参数,但第一个参数需要是表示自身类的cls参数。
4.方法内部
- 如果在@staticmethod中要调用到这个类的一些属性方法,只能直接类名.属性名或类名.方法名。
- 而@classmethod因为持有cls参数,可以来调用类的属性,类的方法,实例化对象等,避免硬编码。
- 如果需要调用带self的方法,必须得先对类进行实例化。
class Test(object): name = 'stephen' """docstring for Test""" def __init__(self, arg=None): super(Test, self).__init__() self.arg = arg def say_hi(self): print('[1]hello wrold') @staticmethod def say_bad(): print('[2]say bad') print('[3]Test.name:%s'%Test.name) # //不需要实例化 Test().say_hi() # //先要实例化,才能调用say_hi() @classmethod def say_good(cls): print('[4]say good') print('[5]cls.name:%s'%cls.name) # //不需要实例化 cls.say_bad() # //不需要实例化 cls().say_hi() # //先要实例化,才能调用say_hi() def main(): Test.say_bad() print("----------------------------------") Test.say_good() if __name__ == '__main__': main()
输出结果
[2]say bad [3]Test.name:stephen [1]hello wrold ---------------------------------- [4]say good [5]cls.name:stephen [2]say bad [3]Test.name:stephen [1]hello wrold [1]hello wrold