python类的__slots__属性、__del__属性、上下文(__enter__和__exit__)、

时间:2022-05-11 12:08:02

常规情况下,类的属性字典是共享的,而实例的字典是独立的。如果一个类的属性较少,但是拥有很多的实例,这些实例的属性字典会占用较多的内存空间。对这样的类来说,为了节省内存空间,可以使用__slots__类变量代替__init__()来存储数据属性。

定义__slots__之后,系统会为实例使用一种更加紧凑的内部表示,即通过一个很小的固定大小的数组来构建,而不是为每一个实例定义一个字典。

__slots__可以是列表、元组、字符串等可迭代对象

定义__slots__之后,类保留有__dict__,但实例没有,取代的是__slots__,且实例不能增加__slots__定义之外的属性。

class animal:
__slots__ = ('name','age')
def eat(self):
print('%s is eating'%self.name) cat=animal()
print(animal.__dict__)#类还是有__dict__的
# print(cat.__dict__)会报错,实例已经没有__dict__字典了
print(cat.__slots__) #('name', 'age'),只返回数据属性名称,不返回对应的值
cat.eat()
#cat.sex='male'会报错,不能为实例增加除了__slots__定义之外的属性了

__del__析构函数,在实例被删除、或者整个python文件执行完毕后自动执行,释放空间。

class test:
def __init__(self,name):
self.name=name
def __del__(self):
print('我执行啦') test1=test('haha')
del test1.name
print('------------')
# ------------
# 我执行啦

__enter__和__exit__

在通过类()对类实例化时,触发类的__enter__方法,操作错误或完毕触发__exit__方法

描述符:在一个类中,定义一个类属性为另一个类的实例,如下在B类中定义类属性x值为A的实例,则A类为描述符。

描述符中可定义__get__方法,获取值时触发,__set__方法,设置值时触发,__delete__删除值时触发。

如果A类中实现了__get__和__set__方法,那么A为数据描述符,如果没有实现__set__方法则为非数据描述符。

class A:
pass
class B:
x=A()

优先级:类属性>数据描述符>实例属性>非数据描述符>找不到