浅谈__slots__
__slots__在python中是扮演属性声明(Attribute Declaration)的角色,当然属性本身不用声明,只需要在使用前赋值即可,但是实际上,属性的应用远非在使用前赋值就行,所以有attribute declaration, attribute accessor,attribute management 等概念,本文对__slots__属性声明做一个学习总结,其他的之后会陆续总结。
(1)Why __slots__?
__slots__用来限制类的实例的属性,优化内存(为每个实例都分配命名空间,如果很多实例被建立起来,然而需要较少的属性,就会很浪费内存空间,所以,python对每个slot 属性在每个实例中都保留了足够空间),防止出现码字错误(typos)等。
但是,它也打破了python核心的动态性质,比如,任何命名通过赋值就可以创建,因此它们应当在非常清楚的真的需要的时候使用,引用python 手册的原话:
Best reserved for rare cases where there are large numbers of instances in a memory-critical application.
(2)__slots__基本用法
__slots__用在新类型类中(New-Style class)是类属性,为列表,其中包含了该类实例的属性,类实例化后,仅允许创建在__slots__列表中的属性,如果定义的属性不在其中,则抛出AttributeError异常。以下代码简单的展示了其基本用法:
>>> class c:
__slots__=['a','b'] >>> x=c()
>>> x.a
Traceback (most recent call last):
File "<pyshell#4>", line 1, in <module>
x.a
AttributeError: a
>>> x.a=1
>>> x.b=2
>>> x.a,x.b
(1, 2)
>>> x.c=8
Traceback (most recent call last):
File "<pyshell#8>", line 1, in <module>
x.c=8
AttributeError: 'c' object has no attribute 'c'
(3)注意事项
(a)__slots__与命名空间字典__dict__
有slots的实例,有些根本就没有__dict__属性,有些实例的属性不在__dict__中。接上面代码,继续...
>>> x.__dict__
Traceback (most recent call last):
File "<pyshell#9>", line 1, in <module>
x.__dict__
AttributeError: 'c' object has no attribute '__dict__'
而没有命名空间字典(namespace dictionary)__dict__,就意味着不可能为实例创建一个不在slots中的属性,怎么办?可以通过把__dict__也写到__slots__中来解决。
>>> class c:
__slots__=['a','b','__dict__'] >>> x=c()
>>> x.a=1;x.b=2;x.c=3
>>> x.a,x.b,x.c
(1, 2, 3)
>>> x.__dict__
{'c': 3}
可以看到可以创建一个x.c,而c不在slots中,在__dict__中。而getattr方法可以对slots中的属性进行访问。接上面代码:
>>> getattr(x,'a');getattr(x,'b')
1
2
(b)在超类(superclass)中的多重__slot__
__slots__是一个类属性,所以子类的slots会继承父类的slots。
>>> class E:
__slots__=['c','d'] >>> class D(E):
__slots__=['a','__dict__'] >>> x=D()
>>> x.a=1;x.c=2;x.d=3;x.e=8
>>> x.a,x.d,x.c,x.e
(1, 3, 2, 8)
浅谈__slots__的更多相关文章
- 浅谈 Fragment 生命周期
版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 源码:AndroidDemo/Fragment 文中如有纰漏,欢迎大家留言指出. Fragment 是在 Android 3.0 中 ...
- 浅谈 LayoutInflater
浅谈 LayoutInflater 版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 源码:AndroidDemo/View 文中如有纰漏,欢迎大家留言指出. 在 Android 的 ...
- 浅谈Java的throw与throws
转载:http://blog.csdn.net/luoweifu/article/details/10721543 我进行了一些加工,不是本人原创但比原博主要更完善~ 浅谈Java异常 以前虽然知道一 ...
- 浅谈SQL注入风险 - 一个Login拿下Server
前两天,带着学生们学习了简单的ASP.NET MVC,通过ADO.NET方式连接数据库,实现增删改查. 可能有一部分学生提前预习过,在我写登录SQL的时候,他们鄙视我说:“老师你这SQL有注入,随便都 ...
- 浅谈WebService的版本兼容性设计
在现在大型的项目或者软件开发中,一般都会有很多种终端, PC端比如Winform.WebForm,移动端,比如各种Native客户端(iOS, Android, WP),Html5等,我们要满足以上所 ...
- 浅谈angular2+ionic2
浅谈angular2+ionic2 前言: 不要用angular的语法去写angular2,有人说二者就像Java和JavaScript的区别. 1. 项目所用:angular2+ionic2 ...
- iOS开发之浅谈MVVM的架构设计与团队协作
今天写这篇博客是想达到抛砖引玉的作用,想与大家交流一下思想,相互学习,博文中有不足之处还望大家批评指正.本篇博客的内容沿袭以往博客的风格,也是以干货为主,偶尔扯扯咸蛋(哈哈~不好好工作又开始发表博客啦 ...
- Linux特殊符号浅谈
Linux特殊字符浅谈 我们经常跟键盘上面那些特殊符号比如(?.!.~...)打交道,其实在Linux有其独特的含义,大致可以分为三类:Linux特殊符号.通配符.正则表达式. Linux特殊符号又可 ...
- 浅谈Angular的 $q, defer, promise
浅谈Angular的 $q, defer, promise 时间 2016-01-13 00:28:00 博客园-原创精华区 原文 http://www.cnblogs.com/big-snow/ ...
随机推荐
- 读《Adaptive Thresholding Using the Integral Image》自适应图像阈值
图像的二值化问题总是一个问题.虽然使用深度学习的方法取得了不小的进展,但是传统的方法还是值得借鉴. 刚好随机游走到这篇文章 挖个07年的坟 地址:http://people.scs.carleton ...
- Codeforces 546 E:士兵的旅行 最大网络流
E. Soldier and Traveling time limit per test 1 second memory limit per test 256 megabytes input stan ...
- iOS大V博客
王巍的博客:王巍目前在日本横滨任职于LINE.工作内容主要进行Unity3D开发,8小时之外经常进行iOS/Mac开发.他的陈列柜中已有多款应用,其中番茄工作法工具非常棒. http://onevca ...
- oracle练习-day04
.什么是PL.PL.普通变量和常量使用) :) :.引用型变量 .记录型变量.条件分支语法:if 条件 .根据输入的年龄判断小于输出未成年人,成年人,以上老年人): .loop循环语法:.输出到的数 ...
- mysq8设置编码utf8
设置mysql默认编码utf8 以及其他配置 系统:centos7 vi /etc/my.cnf #红色部分如果以存在则在他的下方添加 [mysql] default-character-set=ut ...
- Java的SPI机制
目录 1. 什么是SPI 2. 为什么要使用SPI 3. 关于策略模式和SPI的几点区别 4. 使用介绍或者说约定 4.1 首先介绍几个名词 4.2 约定 5. 具体的demo实现 5.1 创建服务提 ...
- sizeof strlen 求char*字符串的长度
sizeof只是求变量所占的字节数,sizeof(char *) = 4字节: strlen(char*) 可以得到整个字符串的长度. 如果是数组vec,那么使用sizeof就可以得到整个数组的所占的 ...
- Intellij IDEA中配置TFS
TFS是微软推出的一款研发过程管理利器,C#阵营的VS里做了默认集成,但是对于Java阵营的Intellij IDEA,需要安装插件并进行相应配置才能使用: 1.打开配置 2.搜索并安装插件 3.配置 ...
- 005.Oracle数据库 , 查询多字段连接合并,并添加文本内容
/*Oracle数据库查询日期在两者之间*/ SELECT PKID , OCCUR_DATE, PKID || ' 曾经沧海难为水 ' ||TO_CHAR( OCCUR_DATE, ' yyyy/m ...
- MYSQL登录及常用命令
1.mysql服务的启动和停止 mysql> net stop mysql mysql> net start mysql 2.登陆mysql mysql> 键入命令mysql -ur ...