面向对象编程——实例属性和类属性(六)
由于python是动态语言,根据类创建的实例可以任意绑定属性。
给实例绑定属性的方法是通过实例变量,或者通过self变量:
class Student(object): def __init__(self,name): self.name = name s = Student('Bob') #添加实例 s.score = 90 #添加属性
但是,如果Student类本身需要绑定一个属性呢?可以直接在class中定义属性,这种属性是类属性,归Student类所有:
>>> class Student(object): ... name = 'student' ... >>> s = Student() #创建实例 >>> print(s.name) #打印name属性,因为实例并没有name属性,所以会继续查找class的name属性 student >>> print(Student.name) #打印类的属性 student >>> s.name = "Michael" #给实例绑定name属性 >>> print(s.name) #实例的属性比类的属性的优先级更高 Michael >>> print(Student.name) #但是类的属性并未消失,仍然可以访问 student >>> del s.name #删除实例的属性 >>> print(s.name) #再次调用实例的属性,会从类中继承 student
从上面的例子可以看出,在编写程序的时候,千万不要对实例属性和类属性使用相同的名字,因为相同名称的实例属性将屏蔽掉所有的类属性,但是当你删除掉实例属性后,再使用相同的名称,访问到的将是类属性。
类的属性可以增删(del)改(重新赋值)查(getattr、__dict__、dir()、直接打印),实例的属性也可以增删改查,如果实在命令行中那只是临时的。但是不建议在外边随便改。
练习:
为了统计学生人数,给累增加一个属性,每创建一个自动加一:
class Student(object): count = 0 def __init__(self,name): self.name = name Student.count +=1
测试
if Student.count != 0: print('测试失败!') else: bart = Student('Bart') if Student.count != 1: print('测试失败!') else: lisa = Student('Bart') if Student.count != 2: print('测试失败!') else: print('Student:',Student.count) print('测试通过!')
结果:
Student: 2
测试通过!
还可以使用__del__统计减的次数。
小结:
实例属性属于各个实例所有,互不干扰;
类舒心属于类所有,所有实例共享一个属性;