目录- 类属性和实例属性
- __dict__方法
- __slots__方法
类属性和实例属性
python里面属性分两种,类属性和实例属性。如何去区分一个属性是类属性还是实例属性?在__init__中带有self的是实例属性,与方法平级的是类属性。
代码语言:javascript复制class Man(object):
gender = 'man'
def __init__(self, name, age):
self.name = name
self.age = age
def behavior(self):
print('hello')
def __str__(self):
return 'name = %s, age = %s'%(self.name, self.age)
这个案例里gender是类属性,name和age是实例属性 。类属性和实例属性在用途上有区别呢?
还是上面那个案例,做到如下操作:
代码语言:javascript复制man1 = Man('ric', 19)
man2 = Man('peter', 20)
# 第一步
Man.gender = 'women'
print(Man.gender)
print(man1.gender)
print(man2.gender)
# 第二步
man1.gender = 'man'
print(Man.gender)
print(man1.gender)
print(man2.gender)
# 第三步
Man.gender = 'unknow'
print(Man.gender)
print(man1.gender)
print(man2.gender)
# 第四步
Man.country = 'china'
print(Man.country)
print(man1.country)
print(man2.country)
# 第五步
man1.education = '本科'
print(man1.education)
print(man2.education)
print(Man.education)
第一步:
women
women
women
第二步:
women
man
women
第三步:
unknow
man
unknow
第四步:
china
china
china
第五步:
本科
为何会输出这些,首先要知道以下几点:
- 类无法访问实例属性,实例可以访问类属性
- 实例无法修改类属性
- 类属性可以动态添加,且实例能访问到这些动态添加的属性
- 实例属性之前互相不影响
- 实例属性和类属性同名时,实例访问的是实例属性
根据以上五点,再来捋捋案例上的输出: 第一步:实例访问的是类属性,所以输出都是women,因为类属性被类自身给修改了 第二步:实例man1给自己绑定了gender实例属性,与类属性同名,所以man1访问的是实例属性 第三步:man2访问的是类属性,man1此时访问的是自己的实例属性 第四步:类动态添加country类属性,实例man1和man2都能访问得到 第五步:实例man1绑定了一个实例属性education,但是并不会影响man2实例和类,所以man2和类Man都抛出了属性缺失异常
__dict__方法
类属性和实例属性都可以通过__dict__来获取
代码语言:javascript复制print(Man.__dict__)
print(man1.__dict__)
输出:
{'__module__': '__main__', 'gender': 'unknow', '__init__': <function Man.__init__ at 0x000001AC9D29D820>, 'behavior': <function Man.behavior at 0x000001AC9D29DA60>, '__str__': <function Man.__str__ at 0x000001AC9D29DAF0>, '__dict__': <attribute '__dict__' of 'Man' objects>, '__weakref__': <attribute '__weakref__' of 'Man' objects>, '__doc__': None, 'country': 'china'}
{'name': 'ric', 'age': 19, 'gender': 'man'}
__slots__方法
默认情况下每个实例都会被分配一个dict用来保存实例属性,但是这样会浪费空间,可以通过__slots__属性来省去给实例分配dict。只有在__slots__中定义的属性名才能被实例添加为属性。且__slots__中的属性不能和类属性同名,不然会报冲突错误。
代码语言:javascript复制class Person(object):
__slots__ = ('country', 'education')
gender = 'man'
def __init__(self):
pass
per = Person()
per.country = 'china'
print(per.country)
per.address = 'beijing' # 错误:添加失败
print(per.address)
由于address属性不在__slots__中,所以给person实例添加address属性会失败。
有了__slots__方法后,实例属性只能通过__slots__来获取属性,类属依然可以通过__dict__方法来获取。
代码语言:javascript复制print(per.__slots__)
print(Person.__dict__)