Python day 8(4) 实例属性和类属性
1 由于Python是动态语言,根据类创建的实例可以任意绑定属性。 给实例绑定属性的方法是通过实例变量,或者通过self
变量:
1 class Student(object): 2 def __init__(self, name): 3 self.name = name 4 5 s = Student('Bob') 6 s.score = 90
2
如果Student
类本身需要绑定一个属性,可以直接在class中定义属性,这种属性是类属性,归Student
类所有:
1 class Student(object): 2 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
当我们定义了一个类属性后,这个属性虽然归类所有,但类的所有实例都可以访问到。
3 在编写程序的时候,千万不要对实例属性和类属性使用相同的名字,因为相同名称的实例属性将屏蔽掉类属性,但是当你删除实例属性后,再使用相同的名称,访问到的将是类属性。
4 习题:
为了统计学生人数,可以给Student类增加一个类属性,每创建一个实例,该属性自动增加:
# -*- coding: utf-8 -*- class Student(object): count = 0 def __init__(self, name): self.name = name Student.count=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('Students:', Student.count) print('测试通过!')
大神分析:
一个类,如果被实例化,就会默认执行一次init()函数, 那么,我们就可以在init()函数里对count属性进行累加。 那我们到底是Student.count+=1 呢? 还是self.count+=1 ? 分析一下, self.count,是实例本身的的属性,一个实例对应一个count, 那这样是无法 统计总数的。 所以必须用类的属性,类的属性是所有实例都能共享访问的。 所以就写Student.count+=1 , 能在init()写出这个Student.count+=1, 是很不容易的,1, 首先,老师的例子,是将类属性直接写在类定义的下边,很容易让人误以为这是死格式,不能写到类中的函数里。2,对于类的实例化,是执行init函数,而不会执行类本身,所以类里边的count=0 是不会执行的,count 是不会在每次类实例化的时候,被初始为0的。