Python之面向对象:对象属性解析:MRO不够用,补充3个方法
off999 2025-07-03 18:49 84 浏览 0 评论
引言
在前面的文章中,我们谈及Python在继承关系,尤其是多继承中,一个对象的属性的查找解析顺序。由于当时的语境聚焦于继承关系,所以只是简要提及了属性解析顺序同方法的解析顺序,而方法的解析顺序,在Python中基于C3算法实现,具体的解析顺序,可以从__mro__属性或者mro()方法中得到。
但是,我们已经介绍了更多的面向对象的内容,尤其在引入了property及属性描述符之后,只是应用MRO似乎有些不够用了。
所以,笔者打算通过两篇文章的篇幅对“对象属性解析”进行一个扩展介绍,从而对属性解析的底层逻辑有一个更加清晰的理解,从而使用Python的面向对象更加得心应手。
本篇文章中,我们首先回顾一下继承语境中,属性解析顺序的内容,然后补充3个关于对象属性解析涉及到的函数/方法,从而进一步剖析属性解析的触发机制。
继承语境中的属性解析顺序
首先,我们先来回顾一下继承语境中的属性解析顺序,对这个解析顺序的理解是基础也是很核心的内容,所以,必须熟练掌握。这个默认的解析顺序如下:
1、查找实例属性:即首先会在实例对象的命名空间(__dict__)中进行属性的查找,如果找到了,直接访问该属性,否则进入下一个查找步骤。
2、查找实例所属类的类属性:如果在实例中未找到相关属性,则会在类的命名空间(__dict__)中进行查找。同样的,如果找到了,则直接访问该类属性,否则进行下一步查找。
3、查找基类属性:如果在实例对象所属的类中未找到,则按照MRO(Method Resolution Order,方法解析顺序)在基类中逐个进行查找,找到了直接访问该属性。如果最终没有找到则报错。
接下来,我们通过代码来进行验证:
class DaGongRen:
count = 0
def __init__(self, name):
self.name = name
DaGongRen.count += 1
class Programmer(DaGongRen):
count = 0
def __init__(self, name, gender):
super().__init__(name)
self.gender = gender
Programmer.count += 1
class ProductManager(DaGongRen):
def __init__(self, name, gender):
super().__init__(name)
self.gender = gender
if __name__ == '__main__':
coder = Programmer('张三', '女')
pm = ProductManager('李四', '女')
print(coder.__dict__)
print(pm.__dict__)
print(Programmer.__dict__)
print(ProductManager.__dict__)
print(DaGongRen.__dict__)
# 实例属性,从实例对象的命名空间也就是__dict__字典中获取
print(coder.name)
print(pm.name)
# 类属性:实例命名空间无该属性,但是实例所属类中有该类属性
print(coder.count)
# 基类属性:实例命名空间没有,实例所属类中也无,基于MRO到基类中查找
print(pm.count)
执行结果:
从代码从执行结果可知:
1、name属性在实例对象的命名空间中有,所以,直接从中获取返回;
2、count属性在实例对象的命名空间中没有,所以需要更进一步查找;
3、coder对应的Programmer类中有count类属性,所以直接返回1;
4、pm对应的ProductManager类中没有count类属性,需要进一步去基类中查找;
5、pm.count最终在DaGongRen的类属性中获取到,返回2。
3个函数/方法
当我们通过实例对象访问属性时,要按照对象的解析顺序进行查找,会涉及到几个函数/方法,而且只是从名称来看很容易混淆的3个函数/方法。
1、getattr(obj, 'attr')内置函数
当我们通过实例对象“点”的方式访问属性时,其实有一个等价的内置函数进行实例对象属性的访问,也就是getattr()。相较于“点”操作符,getattr()函数还提供了属性默认值的功能,也就是说,如果访问的属性不存在,则返回默认值,而不抛出AttributeError。
还是前面的代码,我们通过“点”操作符及内置函数getattr()分别进行属性的访问尝试,可以看到如下执行结果:
2、__getattribute__()魔术方法
当访问实例对象的属性时,首先一定会调用__getattribute__()这个魔术方法进行统一的处理,由这个方法进行属性查找顺序的解析,如果找不到会抛出AttributeError异常,此时,如果定义了__getattr__()方法,则还会调用该方法。
这里暂时不做代码演示,最后会通过一个完整地应用3个函数的代码示例进行说明。
3、__getattr__()魔术方法
该魔术方法只在访问属性失败时被__getattribute__()调用,也就是说可,只有当属性未在实例或类、基类中查找到才会触发。当希望为不存在的属性提供默认的行为,或者动态属性时,可以考虑使用__getattr__()方法。优先于getattr()内置函数的默认值逻辑。
class DaGongRen:
count = 0
def __init__(self, name):
self.name = name
DaGongRen.count += 1
class Programmer(DaGongRen):
count = 0
def __init__(self, name, gender):
super().__init__(name)
self.gender = gender
Programmer.count += 1
def __getattr__(self, item):
print(f"属性{item}不存在")
return None
if __name__ == '__main__':
coder = Programmer('张三', '女')
# 最终通过__getattr__方法返回了None,所以不会走getattr的默认值逻辑
print(getattr(coder, 'age', 18))
# __getattr__方法返回了None,不会抛AttributeError
print(coder.age)
执行结果:
最后,我们来看一下将3个方法/函数放在一起的完整的代码示例:
class DaGongRen:
def __init__(self, name):
self.name = name
class Programmer(DaGongRen):
def __init__(self, name, gender):
super().__init__(name)
self.gender = gender
def __getattribute__(self, item):
print(f"尝试访问属性{item}")
super().__getattribute__(item)
def __getattr__(self, item):
print(f"属性{item}不存在")
return None
class ProductManager(DaGongRen):
def __init__(self, name, gender):
super().__init__(name)
self.gender = gender
def __getattribute__(self, item):
print(f"尝试访问属性{item}")
super().__getattribute__(item)
if __name__ == '__main__':
coder = Programmer('张三', '女')
pm = ProductManager('李四', '女')
# 最终通过__getattr__方法返回了None,所以不会走getattr的默认值逻辑
print("=========以下访问同时有__getattribute__和__getattr__==========")
print(getattr(coder, 'age', 18))
# __getattr__方法返回了None,不会抛AttributeError
print(coder.age)
print("=========以下访问有__getattribute__,无__getattr__==========")
print(getattr(pm, 'age', 18))
print(pm.age)
执行结果:
从代码和执行结果可以看出:
1、访问任何一个属性的时候,首先都会调用__getattribute__()方法,不管属性是否存在,该方法都会被触发。所以,如果需要对所有的属性访问进行统一的处理时,可以在该方法中添加处理逻辑。但是,需要注意的是需要小心规避递归访问的情况,通常应该调用super()的__getattribute__()进行相关的后续属性解析处理,否则可能导致属性解析异常。
2、__getattr__()方法,在属性不存在时,会被__getattribute__()方法调用,且优先级是高于getattr()中的默认值处理逻辑的。
3、如果自定义了__getattriubte__()和__getattr__(),会发现属性的解析顺序中的MRO解析顺序可能会被阻断。所以,通常情况下,不应该随便重写__getattribute__()。
总结
关于实例对象的属性解析,本文回顾了继承中的属性解析属性,并且补充了可以影响属性解析行为的3个函数/方法。
当我们把属性描述符引入时,属性的解析顺序将会更加复杂,关于这部分内容,我们在下一篇文章中继续展开。
感谢您的拨冗阅读,如果本文对您学习Python有所帮助,欢迎点赞、收藏。
相关推荐
- 电脑自带的清理垃圾的工具(电脑自带的清理垃圾的工具叫什么)
-
CCleaner是一款免费的系统优化和隐私保护工具,它的体积小、扫描速度非常快,支持自定义清理规则,增强了应用程序清理范围和效果。CCleaner是Piriform(梨子公司)最著名广受好评的系统清理...
- 如何设置本地连接
-
在“控制面板”中,选择“网络和Internet”>“网络和共享中心”。在左侧窗格中,选择“更改适配器设置”。在“网络连接”窗口中,右键单击“本地连接”,然后选择“属性”。在“本地连接...
- 戴尔官网官方网站(戴尔产品官网)
-
查询步骤如下:1.在戴尔电脑的后盖上找到服务编号,并记录下来。2.之后搜索戴尔官网,在打开的官网界面中点击上方的支持选项,并点击产品支持。3.在打开的产品支持界面中,输入电脑后盖上的服务编号。4.如果...
- 黑鲨u盘重装系统教程(黑鲨u盘重装系统步骤8)
-
U盘重装WIn10系统:1、用【u深度u盘启动盘制作工具】制作u盘启动盘,插入电脑usb接口,设置好开机启动项进入u深度主菜单界面,选择“【02】u深度win8pe标准版(新机器)”并回车,2、在u深...
- 电子邮件免费注册入口(电子邮件在线注册)
-
1.在网页上搜索maiI163邮箱登录,如果有邮箱账号密码的话就直接输入并点击“登录”,没有的话就点击“立即注册”。2.点击“立即注册”后进入页面,输入信息点击“注册”。3.注册成功后就直接搜索登录。...
-
- win7如何快速启动(windows7如何快速启动)
-
打开操作系统运行:输入"cmd"并点击回车:系统命令提示符自动打开:使用方法直接运行start打开一个新的命令提示符窗口:运行start+文件的绝对存储路径打开对应的文件:运行start+文件夹路径打开对应...
-
2025-12-29 13:03 off999
- 怎么升级到win11(怎么升级到win11专业版)
-
Windows11可以在「开始菜单-设置-Windows更新」中进行手动更新。如果您想主动更新,需先确保您的电脑符合Windows11的最低系统要求。接着,打开「Windows更...
- 微信好友误删了怎么加回来(微信好友误删了怎么加回来免费)
-
看到他的评论的话,你可以去你发过的内容里去看看。<br/><br/>好友验证的消息、语音)或者朋友圈内容:<br/>如果你这个朋友喜欢和你在朋友圈聊天的话,你他的手机号也有的话方法添加里输入就可...
-
- access安装包(access安装包怎么安装)
-
要下载并安装MicrosoftAccess,可以按照以下步骤进行操作:1.打开您的电脑的浏览器(如谷歌浏览器、火狐浏览器等)。2.在浏览器的搜索栏中输入"下载MicrosoftAccess"。3.从搜索结果中选择适...
-
2025-12-29 11:51 off999
- 云骑士装机大师官方网站(云骑士装机大师软件下载)
-
就是感觉正规吧,还有就是小白那种的比较多,专业店一忽悠就掏钱做系统了。懂装机的哪有花钱去装系统的不靠谱,因为会造成个人信息的泄露。云骑士装机大师是网络装机系统,在网络上能够实现一键装机,非常的简洁方便...
- 万能钥匙下载免费(安心上网万能钥匙下载免费)
-
行1.使用手机功能表中自带的浏览器上网,直接搜索需要的软件进行下载安装(下载安卓版本格式为apk)。2.使用电脑下载APK格式的安装包,连接数据线传输至手机,操作手机在应用程序-我的文件中找到安装包,...
- 500兆宽带用什么路由器(家用路由器什么牌子好 信号强)
-
1、飞鱼星千兆无线路由器家用2600M双频企业级高速穿墙500M光纤游戏加速VW1900/千兆双频/1900M/大型企业路由器无线500m推荐理由:可以提供企业级别的性能,空旷环境覆盖更广大,...
- xp系统怎么卸载软件(xp怎么卸载程序)
-
1、选中此电脑,点击鼠标右键。2、选择属性点击一下。3、在打开的界面选择控制面板。4、点击程序选项下方的卸载。5、选择要卸载的程序软件,点击鼠标右键。6、点击弹出的选项卸载/更改。7、也可以使用电脑管...
欢迎 你 发表评论:
- 一周热门
-
-
抖音上好看的小姐姐,Python给你都下载了
-
全网最简单易懂!495页Python漫画教程,高清PDF版免费下载
-
Python 3.14 的 UUIDv6/v7/v8 上新,别再用 uuid4 () 啦!
-
飞牛NAS部署TVGate Docker项目,实现内网一键转发、代理、jx
-
python入门到脱坑 输入与输出—str()函数
-
宝塔面板如何添加免费waf防火墙?(宝塔面板开启https)
-
Python三目运算基础与进阶_python三目运算符判断三个变量
-
(新版)Python 分布式爬虫与 JS 逆向进阶实战吾爱分享
-
失业程序员复习python笔记——条件与循环
-
系统u盘安装(win11系统u盘安装)
-
- 最近发表
- 标签列表
-
- python计时 (73)
- python安装路径 (56)
- python类型转换 (93)
- python进度条 (67)
- python吧 (67)
- python的for循环 (65)
- python格式化字符串 (61)
- python静态方法 (57)
- python列表切片 (59)
- python面向对象编程 (60)
- python 代码加密 (65)
- python串口编程 (77)
- python封装 (57)
- python写入txt (66)
- python读取文件夹下所有文件 (59)
- python操作mysql数据库 (66)
- python获取列表的长度 (64)
- python接口 (63)
- python调用函数 (57)
- python多态 (60)
- python匿名函数 (59)
- python打印九九乘法表 (65)
- python赋值 (62)
- python异常 (69)
- python元祖 (57)
