首先值得说明的是staticmethod和classmethod都是python中定义的装饰器,用的时候需要在前面加@
即@staticmethod和@classmethod
翻译过来staticmethod是静态方法, classmethod是类方法
下面是一个简单的例子:
class A(object):
v1 = "I belong to class A"
def func(self, x):
print("executing func(%s,%s)" % (self, x))
@classmethod
def class_func(cls, x):
print("executing class_func(%s,%s)" % (cls, x))
print(cls.v1)
@staticmethod
def static_func(x):
print("executing static_func(%s)" % x)
print(x)
a = A()
a.func(1) #"""对象访问普通方法"""
a.class_func(2) #""" 对象访问类方法"""
A.class_func(3) #""" 类访问类方法"""
a.static_func(4) # """ 对象访问静态方法"""
A.static_func(5) # """ 类访问静态方法"""
#static_func(5)
A.v1
fn = A.func
fn(1,2)
print(A.__dict__)
print(a.__dict__)
"""通过打印法线,a的调用都是去A类中查找的"""
在class A中:
- 没有任何装饰器修饰的函数"func",是属于对象的方法,只能访问实例的其他成员。所以func(self,x)传入的第一个参数是"self",指的是实例本身,即例子中的a。只能通过实例调用。
- classmethod是类的方法,即上例中用@classmethod修饰的函数"class_foo",是属于类的方法,所以class_foo(cls,x)传入的第一个参数是"cls",指类本身,即例子中的A。这个方法是一个类方法,虽然属于类,需要访问类的其他成员,但是不用访问实例的其他成员。并且可以在不把类实例化的前提下,通过类名进行调用,但是值得注意的是,classmethod也可以通过实例调用。典型用途:工厂模式的实现。
- staticmethod是静态方法,即这个方法是一个普通方法,虽然属于类,但是不用访问类和实例的其他成员。并且可以在不把类实例化的前提下,通过类名进行调用。值得注意的是,staticmethod也可以通过实例调用。典型用途:工厂模式的实现。
"""如果Student类本身需要绑定一个属性呢?可以直接在class中定义属性,这种属性是类属性,归Student类所有:
"""
class Student(object):
name = 'Student'
"""当我们定义了一个类属性后,这个属性虽然归类所有,但类的所有实例都可以访问到。来测试一下:"""
>>> class Student(object):
... name = 'Student'
...
>>> s = Student() # 创建实例s
>>> print(s.name) # 打印name属性,因为实例并没有name属性,所以会继续查找class的name属性
Student
>>> print(Student.name) # 打印类的name属性
Student
>>> s.name = 'Michael' # 给实例绑定name属性
>>> print(s.name) # 由于实例属性优先级比类属性高,因此,它会屏蔽掉类的name属性
Michael
>>> print(Student.name) # 但是类属性并未消失,用Student.name仍然可以访问
Student
>>> del s.name # 如果删除实例的name属性
>>> print(s.name) # 再次调用s.name,由于实例的name属性没有找到,类的name属性就显示出来了
Student
从上面的例子可以看出,在编写程序的时候,千万不要对实例属性和类属性使用相同的名字,因为相同名称的实例属性将屏蔽掉类属性,但是当你删除实例属性后,再使用相同的名称,访问到的将是类属性。