python的__get__方法看这一篇就足够了
get类型函数
直接上代码:
class TestMain:
def __init__(self):
print('TestMain:__init__')
self.a = 1 if __name__ == '__main__':
t = TestMain()
print(t.a)
在没有任何get函数的情况下很简单,打印结果是:
TestMain:__init__
但是如果访问一个不存在的属性:
if __name__ == '__main__':
t = TestMain()
print(t.a)
print(t.b) # 访问了一个不存在的属性
结果是:
TestMain:__init__
Traceback (most recent call last): File "C:/Users/sk-leilin/Desktop/mpdp-code-master/test.py", line , in <module>
print(t.b)
AttributeError: 'TestMain' object has no attribute 'b'
可以看见报错了,下载我们来测试一下__getattr__函数:
class TestMain:
def __init__(self):
print('TestMain:__init__')
self.a = 1 def __getattr__(self, item):
print('TestMain:__getattr__')
return 2 if __name__ == '__main__':
t = TestMain()
print(t.a)
print(t.b)
打印结果是:
TestMain:__init__ TestMain:__getattr__
我们仍然访问了一个本来不存在的t.b,为什么这里没有报错呢,因为我们定义了__getattr__函数,而且让它直接返回了2,也就是说,如果定义了这个函数后,访问不存在的属性,会自动调用这个函数作为返回值。
接下来我们看一下__getattribute__这个函数:
class TestMain:
def __init__(self):
print('TestMain:__init__')
self.a = 1 def __getattr__(self, item):
print('TestMain:__getattr__')
return 2 def __getattribute__(self, item):
print('TestMain:__getattribute__')
return 3 if __name__ == '__main__':
t = TestMain()
print(t.a)
print(t.b)
打印结果是:
TestMain:__init__
TestMain:__getattribute__ TestMain:__getattribute__
可以看到,无论是访问存在的t.a还是不存在的t.b,都访问到了__getattribute__这个函数,也就是说,只要定义了这个函数,那么属性的访问,都会走到这个函数里面。
我们在看下面的代码:
class TestMain:
def __init__(self):
print('TestMain:__init__')
self.a = 1 def __getattr__(self, item):
print('TestMain:__getattr__')
return 2 def __getattribute__(self, item):
print('TestMain:__getattribute__')
if item == 'c':
raise AttributeError
return 3 if __name__ == '__main__':
t = TestMain()
print(t.a)
print(t.b)
print(t.c)
我们知道只要定义了__getattribute__函数,就肯定执行这个函数来获取属性,这次我们增加了判断如果访问c这个属性,我们抛出异常,最后的结果是:
TestMain:__init__
TestMain:__getattribute__ TestMain:__getattribute__ TestMain:__getattribute__
TestMain:__getattr__
也就是说,如果__getattribute__抛出了AttributeError异常,那么会继续访问__getattr__函数的。
总结:
- 如果定义了
__getattribute__,那么无论访问什么属性,都是通过这个函数获取,包括方法,t.f()这种也是访问的这个函数,此时这个函数应该放回一个方法,如果像例子中,仍然返回一个数字,你会获得一个TypeError: 'int' object is not callable错误- 只要定义了
__getattribute__方法,不管你访问一个存在的还是不存在的属性,都由这个方法返回,比如访问t.a,虽然a存在,但是只要定义了这个访问,那么就不是访问最开始的a了- 如果
__getattribute__抛出了AttributeError异常,并且定了了__getattr__函数,那么会调用__getattr__这个函数,不论这个属性到底是不是存在- 也就是说属性访问的一个大致优先级是:
__getattribute__>__getattr__>__dict__
单独说一说__get__函数
上面说了__getattribute__和__getattr__,这里单独说一下__get__,因为这个涉及到其它的概念,就是描述器(Descriptor)。
一个类只要实现了
__get__,__set__,__delete__中任意一个方法,我们就可以叫它描述器(descriptor)。如果只定义了__get__我们叫非资料描述器(non-data descriptor),如果__set__,__delete__任意一个/或者同时出现,我们叫资料描述器(data descriptor)。
首先明确一点,拥有这个方法的类,应该(也可以说是必须)产生一个实例,并且这个实例是另外一个类的类属性(注意一定是类属性,通过self的方式产生就不属于__get__范畴了)。
也就是说拥有这个方法的类,那么它的实例应该属于另外一个类/对象的一个属性。 直接看代码吧:
class TestDes:
def __get__(self, instance, owner):
print(instance, owner)
return 'TestDes:__get__' class TestMain:
des = TestDes() if __name__ == '__main__':
t = TestMain()
print(t.des)
print(TestMain.des)
其中TestDes定义了__get__方法,在TestMain中,定义了一个类属性des,是TestDes的一个实例,我们访问t.des或者TestMain.des的时候访问的就是访问了TestDes的__get__方法。
打印结果是:
<__main__.TestMain object at 0x0000022563D5D3C8> <class '__main__.TestMain'>
TestDes:__get__
None <class '__main__.TestMain'>
TestDes:__get__
其中,__get__方法的第一个参数是实际拥有者的实例,如果没有则为None,第二个参数是实际所属的类。
看一下下面的代码:
class TestDes:
def __get__(self, instance, owner):
print(instance, owner)
return 'TestDes:__get__' class TestMain:
def __init__(self):
self.des = TestDes() if __name__ == '__main__':
t = TestMain()
print(t.des)
# print(TestMain.des) #很明显这里会报错
我们通过__init__来产生了一个实例的des属性,这时候,print(t.des)访问的就不是__get__函数了,实际打印结果是:
<__main__.TestDes object at 0x00000165A77ECCF8>
也就是当成一个普通的实例来处理的。
非资料描述器,也就是只有
__get__,不管是类还是实例去访问,默认都获得的是__get__的返回值,但是,如果中间有任何一次重新赋值,那么,这个实例获得的是新的值(对象),已经和原来的描述器完全脱离了关系
资料描述器,比如有__set__方法,后期通过实例对描述器进行赋值,那么访问的是__set__,并且永远关联起来。但是如果通过修改类属性的方式复制,那么也会被重新获取新的值(对象)。
看下面的代码:
class TestDes:
def __get__(self, instance, owner):
print('TestDes:__get__', instance, owner)
return 'TestDes:__get__' class TestMain:
des = TestDes() if __name__ == '__main__':
t = TestMain()
print(t.des)
print(TestMain.des) print() t.des = 1
print(t.des)
print(TestMain.des) print() TestMain.des = 1
print(t.des)
print(TestMain.des)
上面是一个非资料描述器,打印结果是:
TestDes:__get__ <__main__.TestMain object at 0x000002C9BCCF0080> <class '__main__.TestMain'>
TestDes:__get__
TestDes:__get__ None <class '__main__.TestMain'>
TestDes:__get__ TestDes:__get__ None <class '__main__.TestMain'>
TestDes:__get__
具体根据上面的描述行为进行分析,就可以得出结果了。
我们在看一下资料描述器:
class TestDes:
def __get__(self, instance, owner):
print('TestDes:__get__', instance, owner)
return 'TestDes:__get__' def __set__(self, instance, value):
print('TestDes:__set__', instance, value) # 其它代码没有修改
打印结果如下:
TestDes:__get__ <__main__.TestMain object at 0x000002140A46D390> <class '__main__.TestMain'>
TestDes:__get__
TestDes:__get__ None <class '__main__.TestMain'>
TestDes:__get__ TestDes:__set__ <__main__.TestMain object at 0x000002140A46D390>
TestDes:__get__ <__main__.TestMain object at 0x000002140A46D390> <class '__main__.TestMain'>
TestDes:__get__
TestDes:__get__ None <class '__main__.TestMain'>
TestDes:__get__
总结
__getattribute__和__getattr__用于实例访问属性使用,拥有__get__方法的类是只能其实例属于类属性的时候生效- 只要有
__getattribute__,任何属性访问都是这个的返回值,以下都是在__getattribute__不存在或者有AttributeError异常发生的情况下描述的 - 访问不存在的属性,
__getattr__生效 - 访问存在的属性,如果是描述器,描述器生效
- 如果通过实例对描述器进行赋值操作,又有资料和非资料描述器的区分,如果定义了
__set__,那么此方法生效,并且仍然是原始的资料描述器,否则被赋值为新对象 - 描述器赋值如果是通过类的属性方式赋值,而不是类的实例方式赋值,描述器失效
针对描述器的说明: 描述器是被
__getattribute__调用的,如果重写了这个方法,将会阻止自动调用描述器,资料描述器总是覆盖了实例的__dict__, 非资料描述器可能覆盖实例的__dict__。
python的__get__方法看这一篇就足够了的更多相关文章
- Python 变量详解[学习 Python 必备基础知识][看此一篇就够了]
您的"关注"和"点赞",是信任,是认可,是支持,是动力...... 如意见相佐,可留言. 本人必将竭尽全力试图做到准确和全面,终其一生进行修改补充更新. 目录 ...
- Mybatis-Plus常用的查询方法--看这一篇就够了!!!
前言: Mybatis-Plus作为Mybatis的增强,自己封装了很多简单还用的方法,来解脱自己写sql! 对于项目的搭建小编就不在说了,可以参考: SpringBoot+Mybatis-Plus的 ...
- 手写Promise看着一篇就足够了
目录 概要 博客思路 API的特性与手写源码 构造函数 then catch Promise.resolved Promise.rejected Promise.all Promise.race 概要 ...
- 有关Spring事务,看这一篇就足够了
本文将按照声明式事务的五个特性进行介绍: 事务传播机制 事务隔离机制 只读 事务超时 回滚规则 Spring事务传播机制 事务的特性 原子性(Atomicity):事务是一个原子操作,由一系列动作组成 ...
- Java多线程看这一篇就足够了(吐血超详细总结)
进程与线程 进程是程序的一次动态执行过程,它需要经历从代码加载,代码执行到执行完毕的一个完整的过程,这个过程也是进程本身从产生,发展到最终消亡的过程.多进程操作系统能同时达运行多个进程(程序),由于 ...
- java开发环境配置,看这一篇就足够了!
可能平时大家对于安装环境的需求不是那么强烈,但是当你换了一台新电脑时,你就会发现怎么也得花费你几个小时乃至半天一天的时间.故此整理此篇文章,给有需要的小伙伴 注:本文皆win10环境 (1).JDK的 ...
- python面试看这一篇就够了
python-面试通关宝典 有面Python开发方向的,看这一个repo就够啦? 语言特性 1.谈谈对 Python 和其他语言的区别 Python属于解释型语言,当程序运行时,是一行一行的解释,并运 ...
- 看完100篇Python技术精华文章,平均涨薪30%!
一个以技术为立身根基的教育机构做出来的微信号,干货程度会有多高? 马哥Linux运维公众号运营五年,从一开始的定位就是给技术人分享加薪干货的地方.这五年里,公众号运营最重的任务就是做内容.内容并不好做 ...
- 传说中Python最难理解的点,看这完篇就够了
本文转载自简书,作者为菜鸟,感谢作者的辛苦付出. 这不是我第一次学Python入门课,去年.前年我都学过Python入门.所以文章的标题一点都没有标题党的意思.但是整个入门篇还有一个最难的东西没有讲, ...
随机推荐
- PHP开发环境搭建工具有哪些?
对于php开发小白来说搭建一个php运行环境就是一道坎! 因为要做php开发,搭建一个能够运行php网站的服务器环境是第一步,传统的php环境软件非常复杂,好在很多公司开发了一键搭建php安装环境,一 ...
- xxl-job搭建、部署、SpringBoot集成xxl-job
一.搭建xxl-job 1.下载xxl-job代码 码云地址:https://gitee.com/xuxueli0323/xxl-job gitHub地址:https://github.com/xux ...
- C#客户端通过安全凭证调用webservice
怎么解决给XML Web services 客户端加上安全凭据,从而实现调用安全的远程web方法?首先,有远程web服务Service继承自System.Web.Services.Protocols. ...
- 腾讯T8纯手写66个微服务架构设计模式,全部学会真的“变强”了
微服务的概念虽然直观易懂,但“细节是魔鬼”,微服务在实操落地的环节中存在诸多挑战.我们在为企业提供PaaS.人工智能.云原生平台等数字化转型解决方案时也发现,企业实现云原生,并充分利用PaaS能力的第 ...
- Flutter —快速开发的IDE快捷方式
老孟导读:这是老孟翻译的精品文章,文章所有权归原作者所有. 欢迎加入老孟Flutter交流群,每周翻译2-3篇付费文章,精彩不容错过. 原文地址:https://medium.com/flutter- ...
- 用.NET做B/S结构的系统,您是用几种结构来开发,每一层之间的关系以及为什么要这样分层?
表现层(UI):通俗讲就是展现给用户的界面,即用户在使用一个系统的时候他的所见所得. 业务逻辑层(BLL):针对具体问题的操作,也可以说是对数据层的操作,对数据业务逻辑处理. 数据访问层(DAL):直 ...
- 如何提高CSS性能?CSS优化、提高性能提升总汇
如何提高CSS性能,根据页面的加载性能和CSS代码性能,主要表现为: 加载性能 (主要是从减少文件体积,减少阻塞加载,提高并发方面入手),选择器性能,渲染性能,可维护性. 1.尽量将样式写在单独的 ...
- Vue-base64移动端PDF展示
作为一个后端开发,写前端的一些功能也是头大,好在网友强大,网上资源比较多:做一个移动端PDF预览的功能,本来可以通过window.open(),打开的,但是没办法,做后台的小伙伴,传给前端的数据是ba ...
- Python——读取大文件(GB)
最近处理文本文档时(文件约2GB大小),出现memoryError错误和文件读取太慢的问题,后来找到了两种比较快Large File Reading 的方法,本文将介绍这两种读取方法. Prelimi ...
- 查看sudo的history:配置sudolog
sudo 权力很大,但责任更重大! We trust you have received the usual lecture from the local System Administrator. ...