第075讲:类和对象(XVIII)
0. 本节视频https://www.bilibili.com/video/BV1c4411e77t?p=76
1. 温馨提示
如果在学习本节课的过程中遇到问题,可以在这个帖子下方提问哦~
打卡
终于学习到这里了 本节继续讲解描述符,首先通过一个典型的例子说明,描述符只能应用于类属性,而不能应用于对象属性。描述符分为数据描述符和非数据描述符,前者是指实现了__set__()或__delete__()中任意一个方法的描述符,后者是指仅实现了__get__()方法的描述符。当发生属性访问时,数据描述符、实例对象属性、非数据描述符、类属性的优先级从高到低依次递减。事实上,该优先级顺序取决于管理属性获取的__getattribute__()方法的默认实现逻辑,重写该方法有望打破上述访问优先级的“碾压”。
Python3.6为描述符新添加了除__get__()、__set__()、__delete__()之外的第四个魔法方法——__set_name__(),用于捕获被描述符拦截的属性名,可使代码更为优雅,符合Python程序员一贯遵循的简洁易读的编程理念!
学完了这节课,小古比鱼有个小小的疑问:如果一个类仅仅实现了__set_name__(),而没有实现__get__()、__set__()和__delete__(),那么该类还是描述符吗?如果是,那么该类是属于数据描述符呢,还是属于非数据描述符呢? 数据描述符、非数据描述符、优雅编程
class D:
def __get__(self, instance, owner):
print('get~')
class C:
def __init__(self):
self.x = D()
c = C()
print(c.x)# <__main__.D object at 0x0000020FC2EDCFA0>
#描述符只能应用于类属性,而这里把它应用于对象属性,使用不会打印get~
class C:
x = D()
c = C()
c.x# get~
# 数据描述符:实现__set__()和__delete__()中的一个
# 非数据描述符:仅实现__get__()方法
# 当发生属性访问的时候,优先级从高到低依次是:数据描述符 -> 实例对象属性 -> 非数据描述符 -> 类属性
c.x = 'fishc'
print(c.x)# fishc
# D为非数据描述符,优先级小于实例对象属性,所以可以更改实例对象的属性
C.x# get~
class D:
def __get__(self, instance, owner):
print('get~')
def __set__(self, instance, value):
print('set~')
class C:
x = D()
c = C()
c.x = 'FishC'# set~
c.x# get~
print(c.__dict__)# {}
# 数据描述符优先级大于实例对象属性
c.__dict__['x'] = 'FishC'
print(c.__dict__)# {'x': 'FishC'}
c.x# get~
# 拥有__set__()方法后D为数据描述符,访问对象属性依然执行优先级较高的__get__()
# 优先级是定义在__getattribute__()魔法方法实现,因为其管理的是属性的获取
class C:
x = D()
def __getattribute__(self, item):
print('aha~')
c = C()
c.x# aha~
# 没有执行描述符D中的__get__()魔法方法
#__set_name__(self, name, owner):描述符的第四个魔法方法 ptyhon3.6。
# 前边的描述符的魔法方法__get__()、__set__()、__delete__()实现的都是对属性的拦截,在实际开发工作中,我们通过描述符去拦截一个属性,然后做了一些额外的工作,之后通常还要执行相应的访问、赋值或删除等操作。
# 如我们通过描述符拦截了对象的x属性,并进行了合法性验证,判断他的值是不是大于某个值之类的操作,如果符合要求我们仍然要把它写入到实例对象的属性中去,也就是说我们需要在描述符里去操作实例对象的__dict__()字典,这其实是可以做到的,因为instance参数代表的就是描述符所拦截的属性所在的实例对象。
class D:
def __init__(self, name):
self.name = name# name的作用就是记住这个描述符拦截的属性名
def __get__(self, instance, owner):
print('get~')
return instance.__dict__.get(self.name)
def __set__(self, instance, value):
print('set~')
instance.__dict__ = value
class C:
x = D('x')
c = C()
c.x# get~
#__get__()魔法方法拦截成功,此时属性x里没有内容,所以没有返回
print(c.__dict__)# {}
c.x = 250# set~
print(c.__dict__)# {'x': 250}
print(c.x)
# get~
# 250
# class C中代码x = D('x')不够优雅,传入的属性x是以字符串'x'的形式传入。
class D:
def __set_name__(self, owner, name):
self.name = name# name的作用依然是记住这个描述符拦截的属性名
def __get__(self, instance, owner):
print('get~')
return instance.__dict__.get(self.name)
def __set__(self, instance, value):
print('set~')
instance.__dict__ = value
class C:
x = D()# 优雅~~~
c = C()
c.x# get~
print(c.__dict__)# {}
c.x = 250# set~
print(c.__dict__)# {'x': 250}
print(c.x)
# get~
# 250 小古比鱼 发表于 2022-10-10 19:25
本节继续讲解描述符,首先通过一个典型的例子说明,描述符只能应用于类属性,而不能应用于对象属性。描述符 ...
根据定义,数据描述符 插卡。{:10_249:} 小古比鱼 发表于 2022-10-10 19:25
本节继续讲解描述符,首先通过一个典型的例子说明,描述符只能应用于类属性,而不能应用于对象属性。描述符 ...
__set_name__()应该是不能单独定义的,必须跟__set__()一块使用,那就是数据描述符 这节课给我看懵了{:10_280:} CorrineLL 发表于 2023-7-23 17:47
__set_name__()应该是不能单独定义的,必须跟__set__()一块使用,那就是数据描述符
好的。 本帖最后由 h5531465 于 2024-10-1 22:05 编辑
{:5_106:} python繁琐,不好用,总是有边边角角、稀奇古怪的知识点“跳”出来。比c#差得远。
不知道python好用在哪儿。
页:
[1]