Python中的类的方法
Python v3.7.0
静态方法
静态方法实际就是类中的一个普通函数。主要是用来存放逻辑性的代码,逻辑上属于类,但是和类本身没有关系,也就是说在静态方法中,不会涉及到类中的属性和方法的操作。可以理解为,静态方法是个独立的、单纯的函数,它仅仅托管于某个类的名称空间中,便于使用和维护。
通常情况下,静态方法使用@staticmethod
装饰器(或使用staticmethod()
)来声明。如下面的例子:
import time
class TimeTest():
def __init__(self, hour, minute, second):
self.hour = hour
self.minute = minute
self.second = second
@staticmethod
def show_time():
return time.strftime('%H:%m:%s', time.localtime())
if __name__ == '__main__':
# 下面两种方式都可以正确调用
print(TimeTest.show_time())
print(TimeTest(1, 0, 0).show_time())
运行结果为:
20:04:1555158168
20:04:1555158168
在Python 3中,如果一个类的方法没有self参数,不再需要必须声明为静态方法,但是这样的话只能通过类去调用这个方法,如果通过实例对象调用这个方法就会引发异常。
class Animal(object):
# @staticmethod
def greet():
pass
if __name__ == '__main__':
# 以下调用方式会报错
Animal().greet()
异常信息:
TypeError: greet() takes 0 positional arguments but 1 was given
这是因为greet
方法没有声明为静态方法,类实例在调用此方法时,会将self
参数传入到方法中,而方法本身不接受任何参数,从而引发异常。
类方法
定义类方法需要使用装饰器@classmethod
(或使用classmethod()
),传入的第一个参数必须为必须是当前类对象,该参数名一般约定为“cls”,通过它来传递类的属性和方法cls(不能传实例的属性和方法)。
类方法可以通过类直接调用,也可以通过实例直接调用。但无论哪种调用方式,第一个参数一定是类本身。
class Persion(object):
@classmethod
def treasure(cls):
print(__name__, type(cls), cls, sep='||')
if __name__ == '__main__':
# 类对象和实例对象都可以调用
Persion.treasure()
Persion().treasure()
运行结果为:
__main__||<class 'type'>||<class '__main__.Persion'>
__main__||<class 'type'>||<class '__main__.Persion'>
从运行结果可以看出,无论是通过类调用还是实例调用,类方法都能正常工作。且通过打印cls,可以看出cls传入的都是类实例本身。
这里需要注意,如果存在类的继承,那类方法获取的类是类树上最底层的类。如下,Someone类的treasure继承自Persion类,但是从最后的运行结果中可以看到,方法运行时传入的类是Someone,而非其父类。
class Someone(Persion):
pass
if __name__ == '__main__':
Someone.treasure()
Someone().treasure()
运行结果:
<class '__main__.Someone'>
<class 'type'> <class '__main__.Someone'>
实例方法
实例方法第一个参数必须是实例对象本身,该参数名一般约定为“self”,通过它来传递实例对象的属性和方法(也可以传类的属性和方法)。
实例方法在调用的时候,self 是自动传递的,所以不需要我们再处理,但如果使用类对象直接调用实例方法,则需要显式地将实例作为参数传入。
class Test(object):
def __init__(self, name):
self.name = name
def get_gender(self, gender):
return ('{} is a {}').format(self.name, gender)
if __name__ == '__main__':
alex = Test('Alex')
anna = Test('Anna')
# 下面两种方式都可以正确调用
print(alex.get_gender('boy'))
print(Test.get_gender(anna,'girl'))
运行结果为:
Alex is a boy
Anna is a girl
【To be continue...】