9.1 使用_slots_
正常情况下,当我们定义了一个class,创建了一个class的实例后,我们可以给该实例绑定任何属性和方法,这就是动态语言的灵活性。先定义class:
>>> class Animal(object):
pass
然后,尝试给实例绑定一个属性:
>>> an = Animal()
>>> an.name = 'wangda'#动态给实例绑定一个属性
>>> print(an.name)
wangda
还可以尝试给实例绑定一个方法:
>>> def set_age(self,age):
self.age = age
>>> from types import MethodType
>>> an.set_age = MethodType(set_age,an)#给实例绑定一个方法
>>> an.set_age(25)#调用实例方法
>>> an.age
25
但是当我们把一个方法绑定到一个实例之后,其他实例是不起作用的:
>>> ans = Animal()
>>> ans.set_age(12)
Traceback (most recent call last):
File "<pyshell#15>", line 1, in <module>
ans.set_age(12)
AttributeError: 'Animal' object has no attribute 'set_age'
为了给所有实例都绑定方法,可以给class绑定方法,然后所有实例均可调用:
>>> def set_weight(self,weight):
self.weight = weight
>>> Animal.set_weight = set_weight
>>> an.set_weight(10)
>>> an.weight
10
>>> ans.set_weight(25)
>>> ans.weight
25
但是,如果我们想要限制实例的属性怎么办?比如,只允许对Animal实例添加name和age属性。
为了达到限制的目的,Python允许在定义class的时候,定义一个特殊的__slots__变量,来限制该class实例能添加的属性:
>>> class Animal(object):
__slots__ = ('name','age')
>>> an = Animal()
>>> an.name = 'wangda'
>>> an.age = 11
>>> an.weight =25
Traceback (most recent call last):
File "<pyshell#38>", line 1, in <module>
an.weight =25
AttributeError: 'Animal' object has no attribute 'weight'
当我们设置age和name之外的属性时会报错,这就是slots起到的限定的作用
但是,__slots__定义的属性仅对当前类实例起作用,对继承的子类是不起作用的:
>>> class Dog(Animal):
pass
>>> dg = Dog()
>>> dg.age = 11
>>> dg.weight = 25
>>> dg.weight
25
除非在子类中也定义__slots__,这样,子类实例允许定义的属性就是自身的__slots__加上父类的__slots__。