python技术专栏

python属性描述符

2019-08-02  本文已影响0人  huxt

1.对象的自省机制

自省是通过一定的机制查询到对象的内部结构

dir(obj)

  dir(obj)可以获取一个对象所有的属性与方法,返回为列表(仅有属性或方法名称)

  dir()是Python提供的一个API函数,dir()函数会自动寻找一个对象的所有属性(包括从父类中继承的属性和方法)

__dict__

__dict__字典中存储的是对象或类的部分属性,键为属性名,值为属性值

实例对象的__dict__仅存储与该实例相关的实例属性

  类的__dict__存储所有实例对象共享的变量和函数(类属性,方法等),类的__dict__并不包含其父类的属性和方法

dir()和__dict__的区别

  1.dir()是一个函数,返回的是list,仅有属性名和方法名;

  2.__dict__是一个字典,键为属性名,值为属性值;

  3.dir()用来寻找一个对象的所有属性和方法(包括从父类中继承的属性和方法),包括__dict__中的属性和方法,__dict__是dir()的子集;

2.__getattr__、__getattribute__魔法函数

__getattr__ 只是在查找不到属性的时候调用

__getattribute__访问对象的属性(obj.attr)时无条件最先调用该方法,不建议重写该方法

3.属性描述符

  __get__  、   __set__  、   __delete__

实现上述三个魔法函数其中之一即可成为属性描述符,如果只实现__get__称之为非数据属性描述符,只有同时实现__get__和__set__才称之为数据属性描述符

属性描述符

4.属性查找过程

如果user是某个类的实例,那么user.age(以及等价的getattr(user,’age’)),首先调用__getattribute__。如果类定义了__getattr__方法,那么在__getattribute__抛出 AttributeError 的之前就会调用__getattr__,而对于描述符(__get__)的调用,则是发生在__getattribute__内部的。

user = User(), 那么user.age 顺序如下:

(1)如果“age”是出现在User类或其基类的__dict__中(类属性), 且age是data descriptor, 那么调用其__get__方法, 否则

(2)如果“age”出现在user的__dict__中(对象属性), 那么直接返回 obj.__dict__['age'], 否则  -->如果age不是类属性或者User不是数据属性描述符,则去对象属性中查找age

(3)如果“age”出现在User或其基类的__dict__中(类属性)

(3.1)如果age是non-data descriptor,那么调用其__get__方法, 否则

(3.2)返回 class.__dict__['age']

(4)如果User有__getattr__方法,调用__getattr__方法,否则  -->对象属性和类属性中都没有age,则调用User类的__getattr__方法

(5)抛出AttributeError

上一篇下一篇

猜你喜欢

热点阅读