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

目录

    • 类属性和实例属性
    • __dict__方法
    • __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__)



本人是做大数据开发的,在微信上开了个个人号,会经常在上面分享一些学习心得,原创文章都会首发到公众号上,感兴趣的盆友可以关注下哦!
python自学成才之路 类属性和实例属性,__slots__方法_第1张图片
备注:微信公众号搜索‘大数据入坑指南’

你可能感兴趣的:(python)