前面的代码中,我们只给类定义了实例变量,同一个类建立的不同实例的实例属性可以不同。要实现相同类的实例共享属性,可以给类定义类属性和类方法。
1.类属性
顾名思义,类属性就是这个属性是属于类的,而不是属于某一个类实例的。例如以下类定义:
class Foo:
bar = 7
类Foo中没有定义构造方法,但定义了一个类属性。类属性是所有这个类实例的仅有属性,看具体操作:
>>> class Foo:
bar= 7
>>> foo = Foo()
>>> foo2 = Foo()
>>> Foo.bar
7
>>> foo.bar
7
>>> foo2.bar
7
>>> Foo.bar = 0
>>> foo.bar
0
>>> foo2.bar
0
以上是再Python交互环境下运行的结果。可看到,定义Foo类和实例化之后,不管是Foo.bar,还是查看其实例的bar的值都为7。当调用Foo.bar = 0之后,其实例的bar属性也都改变成了0。
注意:如果执行foo.bar = 3,则此时执行的是为foo这个实例添加了一个实例变量bar并赋值为3,并没有改变Foo的类属性bar的值,Foo.bar和foo2.bar返回的是类属性,其值依然是0。而再引用foo.bar则访问的是foo这个实例的实例变量,所以结果为3。
>>> foo.bar =3
>>> foo2.bar
0
>>> foo.bar
3
>>> Foo.bar
0
>>> foo.bar
3
实际上,foo的实例变量bar和类属性bar重名了,引用foo.bar时,优先引用了实例变量。此时我们可以理解为foo的实例变量遮盖了其类属性。而我们一般避免使用foo.bar = 3的形式来改变类的实例,同时也不使用与类属性同名的实例变量。
2.类方法
下面是一个简单的类的代码,定义了一个类方法。
class Foo:
bar = 3
@classmethod
defc_mthd(cls,v):
print(cls.a,v)
以下为实例化类后调用结果和直接用类名的调用结果:
>>> f = Foo()
>>> f.c_mthd(0)
3 0
>>> Foo.c_mthd(0)
3 0
类方法是被classmethod装饰器装饰的,其第一个参数必须是cls,用它来代表类自身。其它参数使用方法与一般方法相同。
注意:类方法中不能使用实例属性,因为调用时可能还没有实例化该类,否则会出错。
3.静态方法
静态方法在定义时就使用staticmethod装饰器,其参数既不用self,也不用cls。从定义形式来看,就像定义一个普通函数一样,如果不希望他被封装在类一起,完全可以在类的外部进行定义一个普通函数,然后在类内的需要的地方直接使用。
class Foo:
bar=3
@staticmethod
defs_mthd(v):
print(Foo.bar,v)
以下是其调用方式:
>>> Foo.s_mthd(0)
3 0
>>> f = Foo()
>>> f.s_mthd(0)
3 0
注意:同类方法相同的是静态方法中也不能使用实例变量,但可以通过类名来使用类属性。但一般情况下,我们定义的静态方法中不应访问类及其实例的属性的。