前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >python自学成才之路 类属性和实例属性,__slots__方法

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

作者头像
我是李超人
发布2020-08-20 20:03:27
8280
发布2020-08-20 20:03:27
举报
文章被收录于专栏:大数据入坑指南

目录
  • 类属性和实例属性
  • __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
第五步:
本科

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

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

根据以上五点,再来捋捋案例上的输出: 第一步:实例访问的是类属性,所以输出都是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__)
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2020/07/12 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 目录
  • 类属性和实例属性
  • __dict__方法
  • __slots__方法
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档