[转载] python __slots__ 详解(上篇)

参考链接: Python的__name __(特殊变量)

python中的new-style class要求继承Python中的一个内建类型,

一般继承object,也可以继承list或者dict等其他的内建类型。

 在python新式类中,可以定义一个变量__slots__,它的作用是阻止在实例化类时为实例分配dict,

 

默认情况下每个类都会有一个dict,通过__dict__访问,这个dict维护了这个实例的所有属性,举例如下: 

 

class base(object):

    var=9 #类变量

    def __init__(self):

        pass

 

b=base()

print b.__dict__

b.x=2 #添加实例变量

print b.__dict__运行结果:

 { }

 {'x': 2}

 可见:实例的dict只保持实例的变量,对于类的属性是不保存的,类的属性包括变量和函数。

 由于每次实例化一个类都要分配一个新的dict,因此存在空间的浪费,因此有了__slots__。

 __slots__是一个元组,包括了当前能访问到的属性。

 当定义了slots后,slots中定义的变量变成了类的描述符,相当于java,c++中的成员变量声明,

 类的实例只能拥有slots中定义的变量,不能再增加新的变量。注意:定义了slots后,就不再有dict。如下: 

 

 

class base(object):

    __slots__=('x')

    var=8

    def __init__(self):

        pass

 

b=base()

b.x=88 #添加实例变量

print b.x

#b.y=99 #无法添加slots之外的变量 (AttributeError: 'base' object has no attribute 'y')

#print b.__dict__ #定义了__slots__后,就不再有__dict__ (AttributeError: 'base' object has no attribute '__dict__')运行结果:

 88

 如果类变量与slots中的变量同名,则该变量被设置为

readonly!!!如下: 

 

 

class base(object):

    __slots__=('y')

    y=22 # y是类变量,y与__slots__中的变量同名

    var=11

    def __init__(self):

        pass

    

b=base()

print b.y

print base.y

#b.y=66 #AttributeError: 'base' object attribute 'y' is read-only运行结果:

 22

 22

 Python是一门动态语言,可以在运行过程中,修改实例的属性和增删方法。一般,任何类的实例包含一个字典__dict__,

 Python通过这个字典可以将任意属性绑定到实例上。有时候我们只想使用固定的属性,而不想任意绑定属性,

 这时候我们可以定义一个属性名称集合,只有在这个集合里的名称才可以绑定。__slots__就是完成这个功能的。 

 

 

class test_slots(object):

    __slots__='x','y'

    def printHello(self):

        print 'hello!'

 

class test(object):

    def printHello(self):

        print 'hello'

 

print dir(test_slots) #可以看到test_slots类结构里面包含__slots__,x,y

print dir(test)#test类结构里包含__dict__

print '**************************************'

ts=test_slots()

t=test()

print dir(ts) #可以看到ts实例结构里面包含__slots__,x,y,不能任意绑定属性

print dir(t) #t实例结构里包含__dict__,可以任意绑定属性

print '***************************************'

ts.x=11 #只能绑定__slots__名称集合里的属性

t.x=12 #可以任意绑定属性

print ts.x,t.x

ts.y=22 #只能绑定__slots__名称集合里的属性

t.y=23  #可以任意绑定属性

print ts.y,t.y

#ts.z=33 #无法绑定__slots__集合之外的属性(AttributeError: 'test_slots' object has no attribute 'z')

t.z=34 #可以任意绑定属性

print t.z 运行结果:

 ['__class__', '__delattr__', '__doc__', '__format__', '__getattribute__', '__hash__', '__init__', '__module__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '

__slots__', '__str__', '__subclasshook__', 'printHello', '

x', 'y']

 ['__class__', '__delattr__', '

__dict__', '__doc__', '__format__', '__getattribute__', '__hash__', '__init__', '__module__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__', 'printHello']

 **************************************

 ['__class__', '__delattr__', '__doc__', '__format__', '__getattribute__', '__hash__', '__init__', '__module__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '

__slots__', '__str__', '__subclasshook__', 'printHello', '

x', 'y']

 ['__class__', '__delattr__', '

__dict__', '__doc__', '__format__', '__getattribute__', '__hash__', '__init__', '__module__', '__new__', '__reduce__', '__reduce_ex__', '__repr__', '__setattr__', '__sizeof__', '__str__', '__subclasshook__', '__weakref__', 'printHello']

 ***************************************

 11 12

 22 23

 34

 

 正如上面所说的,默认情况下,Python的新式类和经典类的实例都有一个

dict来存储实例的属性。这在一般情况下还不错,而且非常灵活,

 乃至在程序中可以

随意设置新的属性。但是,对一些在”编译”前就知道有几个固定属性的小class来说,这个dict就有点浪费内存了。

 当需要创建大量实例的时候,这个问题变得尤为突出。一种解决方法是在

新式类中定义一个__slots__属性。

 __slots__声明中包含若干实例变量,并为每个实例预留恰好足够的空间来保存每个变量;这样Python就不会再使用dict,从而节省空间。

 

【使用memory_profiler模块,memory_profiler模块是在逐行的基础上,测量代码的内存使用率。尽管如此,它可能使得你的代码运行的更慢。使用装饰器@profile来标记哪个函数被跟踪。】 下面,我们看一个例子:

from  memory_profiler import profile

class A(object): #没有定义__slots__属性

    def __init__(self,x):

        self.x=x

 

@profile

def main():

    f=[A(523825) for i in range(100000)]

 

if __name__=='__main__':

    main()运行结果,如下图:

 

 

 第2列表示该行执行后Python解释器的内存使用情况,

第3列表示该行代码执行前后的内存变化。

 在没有定义__slots__属性的情况下,该代码共使用了20.8MiB内存。

 从结果可以看出,内存使用是以MiB为单位衡量的,表示的mebibyte(1MiB = 1.05MB)

from  memory_profiler import profile

class A(object):#定义了__slots__属性

    __slots__=('x')

    def __init__(self,x):

        self.x=x

 

@profile

def main():

    f=[A(523825) for i in range(100000)]

 

if __name__=='__main__':

    main()运行结果,如下图:

 

 

 可以看到,在定义了__slots__属性的情况下,该代码共使用了6.1MiB内存,比上面的20.8MiB节省了很多内存!

 综上所述,在确定了

类的属性固定的情况下,可以

使用__slots__来优化内存。

 提醒:不要贸然进行这个优化,把它用在所有地方。这种做法不利于代码维护,而且只有生成数以千计的实例的时候才会有明显效果。

 

 (完)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/540543.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

委托BegionInvoke和窗体BegionInvoke

委托BegionInvoke是指通过委托方法执行多线程任务,例如: //定义委托成员变量 delegate void dg_DeleAirport(); //指定委托函数 dg_DeleAirport dga AirportBLL.DeleteHistoryTransAirport; //通过BeginInvoke以异步线程方式执行委托函数,可…

图论 弦_混乱的弦

图论 弦Problem statement: 问题陈述: You are provided an input string S and the string "includehelp". You need to figure out all possible subsequences "includehelp" in the string S? Find out the number of ways in which the s…

[转载] Python列表操作

参考链接: Python中的基本运算符 Python列表: 序列是Python中最基本的数据结构。序列中的每个元素都分配一个数字 - 它的位置,或索引,第一个索引是0,第二个索引是1,依此类推; Python有6个序列的…

「原创」从马云、马化腾、李彦宏的对话,看出三人智慧差在哪里?

在今年中国IT领袖峰会上,马云、马化腾、李彦宏第一次单独合影,同框画面可以说很难得了。BAT关心的走势一直是同行们竞相捕捉的热点,所以三位大Boss在这次大会上关于人工智能的见解,也受到广泛关注与多方解读。马云认为机器比人聪明…

python 注释含注释_Python注释

python 注释含注释Python注释 (Python comments) Comments in Python are used to improve the readability of the code. It is useful information given by the programmer in source code for a better understanding of code and logic that they have used to solve the …

C2的完整形式是什么?

C2:核心2 (C2: Core 2) C2 is an abbreviation of "Core 2" or "Intel Core 2". C2是“ Core 2”或“ Intel Core 2”的缩写 。 It is a family of Intels processor which was launched on the 27th of July, 2006. It comprises a series of…

scala特性_Scala | 特性应用

scala特性特性应用 (Trait App) Scala uses a trait called "App" which is used to convert objects into feasible programs. This conversion is done using the DelayedInit and the objects are inheriting the trait named App will be using this function. T…

[转载] Python3中的表达式运算符

参考链接: Python中的除法运算符 1:Python常用表达式运算符 yield 生成器函数send协议 lambda args:expression 创建匿名函数 x if y else z 三元选择表达式(当y为真时,x才会被计算) x or y 逻辑或(仅但x为假时y才会被计算) x and …

字符串矩阵转换成长字符串_字符串矩阵

字符串矩阵转换成长字符串Description: 描述: In this article, we are going to see how backtracking can be used to solve following problems? 在本文中,我们将看到如何使用回溯来解决以下问题? Problem statement: 问题陈述&#xf…

pythonchallenge_level2

level2 地址:http://www.pythonchallenge.com/pc/def/ocr.html。 源码:gitcode.aliyun.com:qianlizhixing12/PythonChallenge.git。 问题:找出页面源码一点提示注释中的稀有字符。 #!/usr/bin/env python3 # -*- coding:UTF-8 -*-# Level 2im…

[转载] python类运算符的重载

参考链接: Python中的运算符重载 alist input().split() blist input().split() n float(input()) class Vector: def __init__(self, x0, y0, z0): # 请在此编写你的代码(可删除pass语句) self.X x self.Y y self.Z z # 代码结束 def __add__(self, other):…

r语言 运算符_R语言运算符

r语言 运算符R语言中的运算符 (Operators in R Language) Generally speaking, an operator is a symbol that gives proper commands to the compiler regarding a specific action to be executed. The operators are used for carrying out the mathematical or logical cal…

[转载] Python基础之类型转换与算术运算符

参考链接: Python中的运算符函数| 1 一、注释 1.注释:对程序进行标注和说明,增加程序的可读性。程序运行的时候会自动忽略注释。 2.单行注释:使用#的形式。但是#的形式只能注释一行,如果有多行,就不方便…

java awt 按钮响应_Java AWT按钮

java awt 按钮响应The Button class is used to implement a GUI push button. It has a label and generates an event, whenever it is clicked. As mentioned in previous sections, it extends the Component class and implements the Accessible interface. Button类用于…

解决“由于应用程序的配置不正确,应用程序未能启动,重新安装应用程序可能会纠正这个问题”...

在VS2005下用C写的程序,在一台未安装VS2005的系统上, 用命令行方式运行,提示: “系统无法执行指定的程序” 直接双击运行,提示: “由于应用程序的配置不正确,应用程序未能启动,重新安…

qgis在地图上画导航线_在Laravel中的航线

qgis在地图上画导航线For further process we need to know something about it, 为了进一步处理,我们需要了解一些有关它的信息, The route is a core part in Laravel because it maps the controller for sending a request which is automatically …

Logistic回归和SVM的异同

这个问题在最近面试的时候被问了几次,让谈一下Logistic回归(以下简称LR)和SVM的异同。由于之前没有对比分析过,而且不知道从哪个角度去分析,一时语塞,只能不知为不知。 现在对这二者做一个对比分析&#xf…

[转载] python学习笔记2--操作符,数据类型和内置功能

参考链接: Python中的Inplace运算符| 1(iadd(),isub(),iconcat()…) 什么是操作符? 简单的回答可以使用表达式4 5等于9,在这里4和5被称为操作数,被称为操符。 Python语言支持操作者有以下几种类型。 算…