说明
- 类属性是类所特有的,所有实例共享这个类属性,类属性经常定义在类的开头,方法的外面
- 实例属性是一个类的实例所特有的,定义在
__init__()里面 - 类属性可以被类和实例调用,实例属性只能被实例调用
来一个栗子A
class A:
x = 1 #类属性
class B(A):
pass
class C(A):
pass
B.x = 2
A.x = 3
print(A.x, B.x, C.x)
#3 2 3
为什么C.x=3, 而不是等于1?
-
类变量和实例变量在内部是通过类对象的字典(
__dict__属性)来处理. 如果在当前类的字典中找不到的话就去它的父类中寻找 -
看下A B C中的
__dict__属性,类C中木有属性x,因此会找父类A中x的值print(A.__dict__) print(B.__dict__) print(C.__dict__) ''' {'__module__': '__main__', 'x': 3, '__dict__': <attribute '__dict__' of 'A' objects>, '__weakref__': <attribute '__weakref__' of 'A' objects>, '__doc__': None} {'__module__': '__main__', '__doc__': None, 'x': 2} {'__module__': '__main__', '__doc__': None} '''
来一个栗子B
class SomeClass:
some_list = [5]
another_list = [5]
def __init__(self, x):
self.some_list = self.some_list + [x]
self.another_list += [x]
some_obj = SomeClass(420)
print(some_obj.some_list) #[5, 420]
print(some_obj.another_list) #[5, 420]
some_obj2 = SomeClass(123)
print(some_obj2.some_list) #[5, 123]
print(some_obj2.another_list) #[5, 420, 123]
为什么some_obj2.another_list 中会含有some_obj中的420?
-
+=运算符会在原地修改可变对象, 而不是创建新对象. 因此, 在这种情况下, 修改一个实例的属性会影响其他实例和类属性 -
查看两个实例对象中的内存地址,
another_list指向同一个内存地址print(id(some_obj.another_list)) print(id(some_obj2.another_list)) ''' 2022218782536 2022218782536 ''' -
我们比较一下类属性和实例属性
print(SomeClass.some_list) #[5] print(some_obj.some_list) #[5, 420] ''' 类属性,在类加载的时候就执行了,它并不会随着实例的调用而改变 '''
本文详细解析了Python中的类属性与实例属性的区别,包括它们的定义位置、调用方式以及内部处理机制。通过具体示例展示了类属性如何被类和实例共享,实例属性如何仅对实例可见。同时,探讨了类属性与实例属性在内存中的存储方式,以及可变对象在不同情况下的行为差异。

被折叠的 条评论
为什么被折叠?



