python 类的__slots__属性

本文介绍了Python中__slots__的使用方法及其限制条件。通过示例解释了如何使用__slots__来限定类实例的属性,并说明了使用__slots__后的实例将不再具备动态创建属性的能力。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

python的_ slots_方法介绍

首先,我们来先看一个例子:

class MySlots:
    def __init__(self):
        self.name = "marin"
        self.age = 18
x = MySlots()
print(x.name)#调用实例变量
print(x.age)
print(x.__dict__)#查看当前实例对象的所有实例变量

输出的结果为:

"marin"
"18"
{"name":"marin","age":18}

此时一切都正常,都比较符合我们的预期,那么再看一下下面的这个例子:

class MySlots:
    def __init__(self):
        self.name = "marin"
        self.age = 18
x = MySlots()
x.weight = 65
print(x.weight)
print(x.__dict__)

输出结果为:

65
{'weight': 65, 'age': 18, 'name': 'marin'}

聪明的看官或许会有疑惑了,MySlots类初始化并没有weight这个属性,为什么实例化对象 x.weight没有报错,还能够创建新的属性.
这其实是在默认情况下,Python用一个dict来存储对象实例的属性。这在一般情况下还不错,而且非常灵活,乃至你在运行时可以随意设置新的属性。

那如果说,我们不想让实例对象动态的创建属性,我们可以这样做:

class MySlots:
    __slots__ = ["name","age"] 
    def __init__(self):
        self.name = "marin"
        self.age = 18
x = MySlots()
x.weight = 65
print(x.weight)
print(x.__dict__)

这个时候如果我们来执行一次这个代码

  File "04_class.py", line 22, in <module>
    x.weight = 65
AttributeError: 'MySlots' object has no attribute 'weight'

这个时候我们已经无法动态的为实例对象创建新的实例变量了
再执行以下

print(x.__dict__)
结果:
AttributeError: 'MySlots' object has no attribute '__dict__'
`

此时,实例对象的__dict__已经不存在了,无法通过__dict__来访问实例变量

总结:一个类有了_slots_方法后,他的实例变量就只能有_slots_绑定的属性.同时,这个类的_dict_方法无法再访问了

注意:
1,不要贸贸然把_slots_用在所有的类里面,这样不利于代码的维护,只用在你认为应该固定属性的地方.

2,_slots_属性的作用范围只在当前类中,无法继承给其他的子类

文章内容仅代表作者想法,如有纰漏或不足之处,欢迎指正

### Python `__slots__` 的用法和作用 #### 定义与基本概念 在Python定义中,可以通过设置特殊属性`__slots__`来限定实例可以拥有的属性名称列表。这不仅有助于减少内存占用,还能提高属性访问速度。 当声明了一个包含`__slots__`变量的之后,该创建的对象将无法动态添加新的成员变量[^1]。 ```python class MyClassWithSlots: __slots__ = ['name', 'identifier'] def __init__(self, name, identifier): self.name = name self.identifier = identifier ``` 上述代码展示了如何通过指定`__slots__`仅允许对象拥有特定的名字(name)和标识符(identifier),任何尝试给此实例增加其他未列于`__slots__`中的属性都会引发异常。 对于不使用`__slots__`的情况: ```python class RegularClassWithoutSlots: def __init__(self, value): self.value = value instance_without_slots = RegularClassWithoutSlots(42) instance_without_slots.new_attribute = "This works fine" print(instance_without_slots.new_attribute) # 输出: This works fine ``` 而如果试图向具有`__slots__`约束的实例分配额外属性,则会抛出错误: ```python my_instance_with_slots = MyClassWithSlots('example_name', 123456) try: my_instance_with_slots.unexpected_attr = True except AttributeError as e: print(e) # 输出似于:'MyClassWithSlots' object has no attribute 'unexpected_attr' ``` #### 使用场景及其优势 - **节省空间**: 对于大量小型实例的应用程序来说非常重要;因为每个实例不再需要字典(dict)存储其状态,而是直接利用固定大小的数据结构。 - **性能提升**: 访问受控字段的速度更快,由于这些字段的位置是预先确定好的,在某些情况下能带来显著效率增益。 - **防止误操作**: 可以有效阻止开发者无意间修改不该被改变的状态信息,增强了代码的安全性和可维护性。 需要注意的是,一旦启用了`__slots__`特性,默认情况下子类不会继承父所设定的槽位集合。因此,若希望保持此功能需显式重申或扩展之。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值