python进阶三(面向对象编程基础)【3-2 python中创建类属型】
python中创建类属性
类是模板,而实例则是根据类创建的对象。
绑定在一个实例上的属性不会影响其他实例,但是,类本身也是一个对象,如果在类上绑定一个属性,则所有实例都可以访问类的属性,并且,所有实例访问的类属性都是同一个!也就是说,实例属性每个实例各自拥有,互相独立,而类属性有且只有一份。
定义类属性可以直接在 class 中定义:
1 class Person(object): 2 address = 'Earth' 3 def __init__(self, name): 4 self.name = name
因为类属性是直接绑定在类上的,所以,访问类属性不需要创建实例,就可以直接访问:
print Person.address # => Earth
对一个实例调用类的属性也是可以访问的,所有实例都可以访问到它所属的类的属性:
1 p1 = Person('Bob') 2 p2 = Person('Alice') 3 print p1.address 4 # => Earth 5 print p2.address 6 # => Earth
任务
请给 Person 类添加一个类属性 count,每创建一个实例,count 属性就加 1,这样就可以统计出一共创建了多少个 Person 的实例。
1 class Person(object): 2 count = 0 3 def __init__(self,name): 4 Person.count = Person.count +1 5 self.name = name 6 7 p1 = Person('Bob') 8 print Person.count 9 10 p2 = Person('Alice') 11 print Person.count 12 13 p3 = Person('Tim') 14 print Person.count
python中类属性和实例属性名字冲突怎么办
修改类属性会导致所有实例访问到的类属性全部都受影响,但是,如果在实例变量上修改类属性会发生什么问题呢?
1 class Person(object): 2 address = 'Earth' 3 def __init__(self, name): 4 self.name = name 5 6 p1 = Person('Bob') 7 p2 = Person('Alice') 8 9 print 'Person.address = ' + Person.address 10 11 p1.address = 'China' 12 print 'p1.address = ' + p1.address 13 14 print 'Person.address = ' + Person.address 15 print 'p2.address = ' + p2.address
结果如下:
Person.address = Earth p1.address = China Person.address = Earth p2.address = Earth
我们发现,在设置了 p1.address = 'China' 后,p1访问 address 确实变成了 'China',但是,Person.address和p2.address仍然是'Earch',怎么回事?
原因是 p1.address = 'China'并没有改变 Person 的 address,而是给 p1这个实例绑定了实例属性address ,对p1来说,它有一个实例属性address(值是'China'),而它所属的类Person也有一个类属性address,所以:
访问 p1.address 时,优先查找实例属性,返回'China'。
访问 p2.address 时,p2没有实例属性address,但是有类属性address,因此返回'Earth'。
可见,当实例属性和类属性重名时,实例属性优先级高,它将屏蔽掉对类属性的访问。
当我们把 p1 的 address 实例属性删除后,访问 p1.address 就又返回类属性的值 'Earth'了:
1 del p1.address 2 print p1.address 3 # => Earth
可见,千万不要在实例上修改类属性,它实际上并没有修改类属性,而是给实例绑定了一个实例属性。