首页 最新 热门 推荐

  • 首页
  • 最新
  • 热门
  • 推荐

第8.20节 Python中限制动态定义实例属性的白名单:__slots__

  • 23-09-22 19:02
  • 2892
  • 12554
blog.csdn.net

一、 引言
按照《第7.10节 Python类中的实例变量定义与使用》、《第7.14节Python类中的实例方法解析》中的介绍,当定义了一个类,并且创建了该类的实例后,可以给该实例动态增加任何属性和方法。但实际上,Python中的类可以控制哪些属性可以增加,这个就类似于一个可以动态增加属性的白名单。这个白名单就定义在类的特殊实例变量__slots__中。
二、 slots
__slots__这个特殊变量在object类中是没有定义的,因此如果使用必须在自定义类中单独定义,并且必须是类变量,不能是实例变量。常规定义的语法如下:
slots = (变量名1,变量名2,…)
slots = [变量名1,变量名2,…]
…

由于__slots__是类变量,因此最好放在类体代码中定义,其中的变量数可以是1到多个。使用第一种方式,当变量是1个时,__slots__自身是字符串类型,当变量是多个时,它是元组类型;使用第二种方式,类变量__slots__类型为列表。实际上__slots__可赋值为字符串、可迭代对象或由实例使用的变量名构成的字符串序列,任何非字符串可迭代对象都可以被赋值给 slots。映射也可以被使用,在此不展开介绍每种赋值方式。
三、 __slots__的作用

  1. slots 允许开发人员显式地声明限定的数据成员(例如特征属性),禁止未声明的成员动态加入;
  2. 这个 slots 会为已声明的变量保留空间,并阻止自动为每个实例创建 dict 和 __weakref_特殊变量(除非是在 slots 中显式地声明或是在父类中可用)。
  1. 而当继承自一个未定义 slots 的类时,实例的 dict 和 weakref 属性将总是可访问;
  2. 没有 dict 变量,实例就不能给未在 slots 定义中列出的新变量赋值。尝试给一个未列出的变量名赋值将引发 AttributeError。新变量需要动态赋值,就要将 ‘dict’ 加入到 slots 声明的字符串序列中。
  3. 如果未给每个实例设置 weakref 变量,定义了 slots 的类就不支持对其实际的弱引用。如果需要弱引用支持,就要将 ‘weakref’ 加入到 slots 声明的字符串序列中。
  1. 使用__slots__相比使用 dict 方式可以显著地节省空间。 属性查找速度也可得到显著的提升;
  2. slots 声明的作用不只限于定义它的类。在父类中声明的 slots 在其子类中同样可用。不过,子类将会获得 dict 和 weakref 除非它们也定义了 slots (其中应该仅包含对任何 额外 名称的声明位置)。
  3. 非空的 slots 不适用于派生自“可变长度”内置类型例如 int、bytes 和 tuple 的派生类;
  4. __ slots__只能在类体代码中赋值,赋值后:
  1. 不能通过类体外使用“实例名.__ slots__”方式重新赋值,Python会报该属性只读;
  2. 不能在实例方法中和类体外代码中使用“类名.__ slots__” 方式赋值,赋值时不会报错,但__ slots__不起作用,如实例中__dict__会自动创建,如果类体中原来已经定义了__slots__,在实例方法中修改__slots__,可以修改,但Python还是只允许类体中定义的__ slots__中限定的属性添加。
  1. 定义了__ slots__后,如果在代码中定义__ slots__外的实例变量,则会报AttributeError错误;
  2. __ slots__定义以后,在实例方法含构造方法中都不能新定义__ slots__外的其他实例变量。

四、 三个案例

  1. 源代码如下:
    #案例1:#使用实例方法来定义类变量Vehicle.__slots__没有作用
class Vehicle(): 
   def __init__(self,power):
       self.power = power
       Vehicle.__slots__ = ['power']
v=Vehicle('人力')
v.wheelcount=4 #加一个实例变量不会拦截
v.__dict__  #服务字典中的自定义属性成功
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7

#案例2:#使用类体来定义类变量__slots__可以正常发挥作用

class Vehicle(): 
   __slots__ = ['power']
   def __init__(self,power):
       self.power = power
       
v=Vehicle('人力')
v.wheelcount=4 #加一个实例变量会拦截
v.__dict__  #没有字典属性      
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8

#案例3:#定义类变量__slots__后在类体外修改,可以修改但修改不起作用

class Vehicle(): 
   __slots__ = ['power','weight']
   def __init__(self,power):
       self.power = power
       
v=Vehicle('人力')
Vehicle.__slots__ = ['wheelcount','oilcost']
Vehicle.__slots__ #显示赋值被修改
v.wheelcount=4 #加一个实例变量会拦截
v.__dict__  #没有字典属性
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  1. 执行截屏:

本节介绍了特殊变量__slots__的定义和使用以及注意事项,并举例进行了说明,通过介绍可以知道,__slots__相当于是一个实例变量的白名单。

老猿Python,跟老猿学Python!
博客地址:http://iyenn.com/index/link?url=https://blog.csdn.net/LaoYuanPython

请大家多多支持,点赞、评论和加关注!谢谢!

文章知识点与官方知识档案匹配,可进一步学习相关知识
Python入门技能树首页概览333596 人正在系统学习中
老猿Python
微信公众号
专注Python相关语言、图像音视频处理、AI
注:本文转载自blog.csdn.net的LaoYuanPython的文章"https://blog.csdn.net/LaoYuanPython/article/details/94890048"。版权归原作者所有,此博客不拥有其著作权,亦不承担相应法律责任。如有侵权,请联系我们删除。
复制链接
复制链接
相关推荐
发表评论
登录后才能发表评论和回复 注册

/ 登录

评论记录:

未查询到任何数据!
回复评论:

分类栏目

后端 (14832) 前端 (14280) 移动开发 (3760) 编程语言 (3851) Java (3904) Python (3298) 人工智能 (10119) AIGC (2810) 大数据 (3499) 数据库 (3945) 数据结构与算法 (3757) 音视频 (2669) 云原生 (3145) 云平台 (2965) 前沿技术 (2993) 开源 (2160) 小程序 (2860) 运维 (2533) 服务器 (2698) 操作系统 (2325) 硬件开发 (2492) 嵌入式 (2955) 微软技术 (2769) 软件工程 (2056) 测试 (2865) 网络空间安全 (2948) 网络与通信 (2797) 用户体验设计 (2592) 学习和成长 (2593) 搜索 (2744) 开发工具 (7108) 游戏 (2829) HarmonyOS (2935) 区块链 (2782) 数学 (3112) 3C硬件 (2759) 资讯 (2909) Android (4709) iOS (1850) 代码人生 (3043) 阅读 (2841)

热门文章

101
推荐
关于我们 隐私政策 免责声明 联系我们
Copyright © 2020-2024 蚁人论坛 (iYenn.com) All Rights Reserved.
Scroll to Top