python自学成才之路 类属性和实例属性,__slots__方法

时间:2022-07-23
本文章向大家介绍python自学成才之路 类属性和实例属性,__slots__方法,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。

目录

类属性和实例属性

python里面属性分两种,类属性和实例属性。如何去区分一个属性是类属性还是实例属性?在__init__中带有self的是实例属性,与方法平级的是类属性。

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是实例属性 。类属性和实例属性在用途上有区别呢?

还是上面那个案例,做到如下操作:

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
第五步:
本科

为何会输出这些,首先要知道以下几点:

  1. 类无法访问实例属性,实例可以访问类属性
  2. 实例无法修改类属性
  3. 类属性可以动态添加,且实例能访问到这些动态添加的属性
  4. 实例属性之前互相不影响
  5. 实例属性和类属性同名时,实例访问的是实例属性

根据以上五点,再来捋捋案例上的输出: 第一步:实例访问的是类属性,所以输出都是women,因为类属性被类自身给修改了 第二步:实例man1给自己绑定了gender实例属性,与类属性同名,所以man1访问的是实例属性 第三步:man2访问的是类属性,man1此时访问的是自己的实例属性 第四步:类动态添加country类属性,实例man1和man2都能访问得到 第五步:实例man1绑定了一个实例属性education,但是并不会影响man2实例和类,所以man2和类Man都抛出了属性缺失异常

__dict__方法

类属性和实例属性都可以通过__dict__来获取

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__中的属性不能和类属性同名,不然会报冲突错误。

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__方法来获取。

print(per.__slots__)
print(Person.__dict__)