python槽怎么用_python 特性slots槽详解

槽(slots)可以使用__slots_属性来为自定的类设置以一个静态属性列表,并在类的每个实例中跳过__dict__字典的创建过程,可以提高访问速度,节省内存消耗

由于Python是动态语言,任何实例在运行期都可以动态地添加属性。

如果要限制添加的属性,例如,Student类只允许添加 name、gender和score 这3个属性,就可以利用Python的一个特殊的__slots__来实现

class Student(object):

__slots__ = ('name', 'gender', 'score')

def __init__(self, name, gender, score):

self.name = name

self.gender = gender

self.score = score

创建实例,并向其中新增属性

>>> s = Student('Bob', 'male', 59)

>>> s.name = 'Tim' # OK

>>> s.score = 99 # OK

>>> s.grade = 'A'

新增属性grade执行后,抛异常【AttributeError: 'student' object has no attribute 'gride'】

当一个Student类定义了__slots__ = ('name', 'age'),Studen.name就是一个有__get__和__set__方法的member_descriptor

__slots__的目的是限制当前类所能拥有的属性,如果不需要添加任意动态的属性,使用__slots__也能节省内存。

__slots__两大特性

更快的属性访问速度

默认情况下,访问一个实例的属性是通过访问该实例的__dict__来实现的。如访问s.name就相当于访问s.__dict__['name']。为了便于理解,我粗略地将它拆分为四步:s.name  >> s.__dict__ >>. s.__dict__['name'] >>. 结果

从__slots__的实现可以得知,定义了__slots__的类会为每个属性创建一个描述器。访问属性时就直接调用这个描述器。在这里我将它拆分为三步:b.x >>. member decriptor >>. 结果

我在上文提到,访问__dict__和描述器的速度是相近的,而通过__dict__访问属性多了s.__dict__['name']字典访值一步(一个哈希函数的消耗)。由此可以推断出,使用了__slots__的类的属性访问速度比没有使用的要快。下面用一个例子验证:

from timeit import repeat

class A(object): pass

class B(object): __slots__ = ('x')

def get_set_del_fn(obj):

def get_set_del():

obj.x = 1

obj.x

del obj.x

return get_set_del

a = A()

b = B()

ta = min(repeat(get_set_del_fn(a)))

tb = min(repeat(get_set_del_fn(b)))

print("%.2f%%" % ((ta/tb - 1)*100))

在本人电脑上测试速度有0-25%左右的提升。

减少内存消耗

python内置的字典本质是一个哈希表,它是一种用空间换时间的数据结构。为了解决冲突的问题,当字典使用量超过2/3时,Python会根据情况进行2-4倍的扩容。由此可预见,取消__dict__的使用可以大幅减少实例的空间消耗。

关于slots的继承问题

在一般情况下,使用slots的类需要直接继承object,如class Foo(object): __slots__ = ()

在继承自己创建的类时,我根据子类父类是否定义了__slots__,将它细分为三种情况:(暂未列出多父类的情况)

父类有,子类没有:子类的实例还是会自动创建__dict__来存储属性,不过父类__slots__已有的属性不受影响。

父类没有,子类有:虽然子类取消了__dict__,但继承父类后它会继续生成。同上面一样,__slots__已有的属性不受影响。

父类有,子类有:只有子类的__slots__有效,访问父类有子类没有的属性依然会报错

因此:为了正确使用__slots__,最好直接继承object。如有需要用到其他父类,则父类和子类都要定义slots,还要记得子类的slots会覆盖父类的slots。

除非所有父类的slots都为空,否则不要使用多继承。

在特殊情况下,可以在__slots__里添加__dict__来获取与普通实例同样的动态特性。

当一个类需要创建大量实例时,可以使用__slots__来减少内存消耗。如果对访问属性的速度有要求,也可以酌情使用。另外可以利用slots的特性来限制实例的属性。而用在普通类身上时,使用__slots__后会丧失动态添加属性和弱引用的功能,进而引起其他错误,所以在一般情况下不要使用它。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
`__slots__` 是 Python 中的一个特殊属性,它允许你显式地指定一个类的实例属性。通过使用 `__slots__`,你可以限制一个类实例可以存在的属性集合,从而优化内存使用和访问速度。 在 Python 中,每个类实例都有一个字典(`__dict__`)用于存储实例的属性。这使得你可以在运行时动态地给实例添加、修改或删除属性。然而,对于拥有大量实例的类,使用字典来存储属性可能会占用大量内存,并且属性查找会比较慢。 使用 `__slots__`,你可以预先声明一个固定的属性集合,从而避免了使用字典来存储实例的属性。通过这种方式,每个实例只会使用固定大小的内存,属性查找也变得更快。 要使用 `__slots__`,你可以在类定义中定义一个包含属性名称的元组。这些属性名称将成为实例的属性,而且除了这些属性之外,任何其他的属性都不能被添加到实例中。以下是一个示例: ```python class MyClass: __slots__ = ('attribute1', 'attribute2') def __init__(self, value1, value2): self.attribute1 = value1 self.attribute2 = value2 # 创建一个类实例 obj = MyClass(10, 20) # 访问实例属性 print(obj.attribute1) # 输出: 10 print(obj.attribute2) # 输出: 20 # 尝试添加新属性 obj.attribute3 = 30 # 抛出 AttributeError:'MyClass' object has no attribute 'attribute3' ``` 在上面的例子中,`MyClass` 类的实例只能具有 `attribute1` 和 `attribute2` 这两个属性。如果你尝试添加一个不在 `__slots__` 中的新属性,将会引发 `AttributeError` 异常。 需要注意的是,使用 `__slots__` 有一些限制和注意事项: - `__slots__` 只对当前类的实例有效,子类不受影响。 - 每个类实例仍然会有一个 `__dict__` 属性,但它只是一个空字典,不会包含实例的属性。 - 使用 `__slots__` 可能会导致一些特殊方法(例如 `__weakref__`)无法正常工作。 - 使用 `__slots__` 可能会限制动态性和灵活性。只有在确实需要优化内存和属性访问速度时才应该使用它。 希望这个解释能够帮到你!如果你还有其他问题,请随时提问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值