关于Python类属性与实例属性的讨论
标题名字有点长。
之所以想写这个文章是因为碰巧看到网上一篇关于Pyhon中类属性及实例属性区别的帖子。因为我之前也被这个问题困扰过,今天碰巧看到了这篇帖子,发现帖子的作者只是描述了现象,然后对原因的解释比较含糊,并没有从根本上解释这个问题,所以才想写一下我对这个问题的想法。
性子急的可以直接跳到最后看总结。
原帖子地址
问题描述
为了方便对比,我还是使用原帖子的例子:
class AAA(): aaa = 10 # 情形1
obj1 = AAA()
obj2 = AAA()
print obj1.aaa, obj2.aaa, AAA.aaa # 情形2
obj1.aaa += 2
print obj1.aaa, obj2.aaa, AAA.aaa # 情形3
AAA.aaa += 3
print obj1.aaa, obj2.aaa, AAA.aaa
情形1的结果是:10 10 10
;
情形2的结果是:12 10 10
;
情形3的结果是:12 13 13
;
首先为什么会有这个问题呢?
因为aaa
属性被称为类属性,既然是类属性,那么根据从C++/Java
这种静态语言使用的经验来判断,类属性应该是为其实例所共享的。很自然的,既然是共享关系,那么从类的层次改变aaa
的值,自然其实例的aaa
的值也要跟着变化了。
可是情形3的情况却说明,上面的说法是错的。
错哪里呢?
要从Python的类属性讲起
Python中类属性的含义
Python属于动态强类型的语言,在很多地方和静态语言不同,因此,不能把静态语言的规则套到动态语言上来。其中,类属性就是一个很好的例子。
Python中属性的获取
对于属性,我们通常采用类.属性或实例.属性的形式调用。
例如上例中的AAA.aaa
属于类.属性形式,obj1.aaa
属于实例.属性的形式
Python中属性的设置
对于属性的设置我们通常采用类.属性 = 值或实例.属性 = 值的形式
例如obj1.aaa = 3
上例中obj1.aaa += 2
等价于obj1.aaa = obj1.aaa + 2
,这句话包含了属性获取及属性设置两个操作
OK,重点来了,Python中属性的获取和设置的机制与静态语言是不同的,正是背后机制的不同,导致了Python中类属性不一定是为其实例所共享的
Python中属性查找机制
Python中属性的获取存在一个向上查找机制,还是拿上面的例子做说明:
Python中一切皆对象,AAA
属于类对象,obj1
属于实例对象,从对象的角度来看,AAA
与obj1
是两个无关的对象,但是,Python通过下面的查找树建立了类对象AAA
与实例对象obj1
、obj2
之间的关系。
如图所示
AAA|-----| | obj1 obj2
(图画的不好,见谅 -.-!!!)
当调用AAA.aaa
时,直接从AAA
获取其属性aaa
。
但是情形1中调用obj1.aaa
时,Python按照从obj1
到AAA
的顺序由下到上查找属性aaa
。
值得注意的这时候obj1
是没有属性aaa
的,于是,Python到类AAA
中去查找,成功找到,并显示出来。所以,从现象上来看,AAA
的属性aaa
确实是共享给其所有实例的,虽然这里只是从查找树的形式模拟了其关系。
Python中的属性设置
原帖子的作者也指出问题的关键在于情形2中obj1.aaa += 2
。
为什么呢?
上面我们指出obj.aaa += 2
包含了属性获取及属性设置两个操作。即obj1.aaa += 2
等价于obj1.aaa = obj1.aaa + 2
。
其中等式右侧的obj.aaa
属于属性获取,其规则是按照上面提到的查找规则进行,即,这时候,获取到的是AAA
的属性aaa
,所以等式左侧的值为12
。
第二个操作是属性设置,即obj.aaa = 12
。当发生属性设置的时候,obj1
这个实例对象没有属性aaa
,因此会为自身动态添加一个属性aaa
。
由于从对象的角度,类对象和实例对象属于两个独立的对象,所以,这个aaa
属性只属于obj1
,也就是说,这时候类对象AAA
和实例对象aaa
各自有一个属性aaa
。
那么,在情形3中,再次调用obj1.aaa
时,按照属性调用查找规则,这个时候获取到的是实例对象obj1
的属性aaa
,而不是类对象AAA
的属性aaa
。
对问题探讨的总结
到这里就可以完满解释上面的问题:
1. Python中属性的获取是按照从下到上的顺序来查找属性;
2. Python中的类和实例是两个完全独立的对象;
3. Python中的属性设置是针对对象本身进行的;
对情形1的解释
因为Python中的属性获取是按照从下到上的顺序来查找的,所以在情形1:
obj1 = AAA()
obj2 = AAA()
实例对象obj1
和obj2
不存在属性aaa
。
证明如下:
>>> obj1.__dict__
{}
>>> obj2.__dict__
{}
所以,此时,obj1.aaa, obj2.aaa, AAA.aaa
实质上都是指AAA.aaa
。因此,输出同样的结果。
对情形2的解释
因为Python中的类和实例是两个完全独立的对象且Python中的属性设置是针对对象本身进行的,所以在情形2:
obj1.aaa += 2
实质上是对实例对象obj1
设置了属性aaa
,并赋值为12
。证明如下:
>>> obj1.aaa = 3
>>> obj1.__dict__
{'aaa': 3}
>>> obj2.__dict__
{}
因此,再次调用obj1.aaa
时,将获取到的是实例对象obj1
的属性aaa
,而不是类对象AAA
的属性aaa
。而对于实例对象obj2
,由于其并没有属性aaa
,所以调用obj2.aaa
时,获取到的是AAA
的属性aaa
。
对情形3的解释
顺利理解了前两个情形,那么第3个情形就很容易了,改变AAA
的属性aaa
只能影响到类对象AAA
和实例对象obj2
,不能影响obj1
,因为,obj1
存在aaa
,在获取时,不会获取到AAA
的属性。
写在最后的话
问题本身很简单,但是通过对这个问题的探讨,可以深入理解Python作为一个动态语言,在OOP的机制上与静态语言的差别。
最关键的地方在于两点:
1. 理解Python是如何利用查找树的机制来模仿类及实例之间的关系;
2. 理解动态语言是可以动态设置属性的
关于Python类属性与实例属性的讨论相关推荐
- Python 类—类属性(私有属性、公有属性、实例属性、局部变量)类方法(实例方法、静态方法)
1. 创建类 类是对某个对象的定义,它包含有关对象动作方式的信息,包括它的名称.方法.属性和事件.类不存在于内存中,因此它本身并不是对象.当程序运行需要引用类的代码时,就会在内存中创建一个类的新实例, ...
- python实例属性与类属性_Python中的类属性和实例属性引发的一个坑-续
上篇文章Python 中的类属性和实例属性,我们探讨了类属性和实例属性引发的一个小坑.总结了类属性和实例属性的区别如下:类属性可以被类和实例调用,实例属性只能被实例调用 类属性不会随着实例的调用而改变 ...
- Python类属性和实例属性分别是什么?
实例属性即同一个类的不同实例,其值是互不关联的,也不会互相影响的,定义时使用"self.属性名",调用时也使用"self.属性名". 类属性则是同一个类的所有实 ...
- python的实例属性和静态属性表_Python:类属性,实例属性,私有属性与静态方法,类方法,实例方法...
类属性可以直接在方法中赋值,但 一.调用该属性时,必须先调用该方法: 二.与 init 方法区别是,init方法只初始化一次,初始化完成后作为全局变量.在方法中赋值属性,每次调用该方法,该属性初始化一 ...
- python类属性和实例属性_Python 类属性与实例属性,类对象与实例对象用法分析...
本文实例讲述了Python 类属性与实例属性,类对象与实例对象用法.分享给大家供大家参考,具体如下: demo.py(类属性,所有实例对象共用类属性): # 定义工具类 继承object是为了兼容py ...
- Python OOP:面向对象三大特性(封装、继承、多态)、类属性和实例属性、类方法@classmethod、静态方法@staticmethod、函数和类callable、__call__
一.面向对象三大特性 二.多态 定义:多态是⼀种使⽤对象的⽅式,⼦类重写⽗类⽅法,调⽤不同⼦类对象的相同⽗类⽅法,可以产⽣不同的执⾏结果. 一个父类,可以继承给多个子类,每个不同的子类可以创建多个对象 ...
- 【python】类属性以及实例属性、实例方法的介绍
1. 类的组成 class Demo():var = 'demo' # 类属性,公有__num = 100 # 类属性,私有def __init__(self,name='muzi',age=18): ...
- python类中的属性分为类属性和实例属性两种_python从入门到大神---1、初始化实例、类属性、方法...
python从入门到大神---1.初始化实例.类属性.方法 一.总结 一句话总结: 方法不加括号是代码段:感觉python方法和js,php很类似,不加括号是代码段,加括号变成方法,比如f,f() 1 ...
- Python 类属性与实例属性
首先我们简要说下类属性与实例属性在概念上的不同之处: 类属性是在类中定义的属性,它是和这个类所绑定的,这个类中的所有对象都可以访问.访问时可以通过类名来访问,也可以通过实例名来访问. 实例属性是与类的 ...
最新文章
- 寻找连通域算法_【车牌识别算法】
- 用 Navicat for Oracle 管理 Oracle10g/11g 数据库
- AI 盯上了外包司机,看后视镜就被扣分,奖金拜拜!
- 【tensorflow】tf.nn.conv2d的使用
- linux 6.3 vnc安装包,CentOS 6.3安装和配置VNC
- 直接插入排序比较次数_程序员必须要会的直接插入排序算法
- CSS three column layout
- 编写Java程序,在屏幕上显示带标题的窗口,并添加一个按钮。当用户单击按钮时,结束程序。
- 360度动态漫游技术(html格式)
- (day 21 - 辅助栈加模拟)剑指 Offer 31. 栈的压入、弹出序列
- SCADA数采HAAS数控系统现场实操步骤
- Android Studio中运行Android模拟器
- python定义矩形类_python类矩形新类型
- 纪一次笔试之行—燃烧的远征
- Java语言 Timer 定时器的四种使用方式
- linux shell 10进制转16进制
- Alfred Workflow 豆瓣插件开发
- go-zero创建报required as: github.com/tal-tech/go-zero
- HTTP代理与SOCKS代理的区别
- OpenCV和Python进行SIFT算法——两张图片的关键点匹配
热门文章
- python基础知识点大全-【python基础学习】基础重点难点知识汇总
- windows错误:Failed to import pydot. You must install pydot and graphviz for `pydotprint` to work.
- Opengl-模型(告别箱子加载模型)
- php7.2 的好处,PHP 7.2 中弃用的功能
- oracle 如何获取表的主键列名,如何获取表的所有列名
- java ThreadLocal用法及原理
- 七天学会ASP.NET MVC(七)——创建单页应用
- win7命令行 端口占用 查询进程号 杀进程
- 图像投影特征图的波峰波谷查找的相关原理及利用差分遍历法查找波峰的OpenCV代码
- oracle安装报错emca,求助:oracle 安装问题