文档章节

Python的mutable值碎碎念

守恒的猫
 守恒的猫
发布于 2017/04/10 10:16
字数 658
阅读 6
收藏 0

如下案例:

>>> def test(a, b=[]):
...     b.append(a)
...     return b

>>> test(1)
[1]
>>> test(2)
[1, 2]
>>> test(3)
[1, 2, 3]

 

是由于对于python的可变类型list,set,dict等,它的关键字变量默认值是在函数定义的时候存在func_defaults里的,又因为可变类型是传引用,因此每一次调用都是变化的。

 

[另一方面,在嵌套函数中,python2的内层函数无法获取外层函数不可变变量,但可以访问可变变量,因此通常这样写

def test():

    a = [1]

    def t():

        print a[0]

    t()

把变量放在一个mutable里]X

这里有一个严重失误,像这样的闭包,python内存函数是可以访问外层的变量的,但对immutable变量不允许修改,外面有个a=1,里面可以print a,可以a=2,但不允许a += 2,之前的“无法获取”是错误的,谨记!!!

——————————————————————————————————————————————

说到mutable,一个值得探讨的问题就是copy和deepcopy。通常来说一个浅拷贝只拷贝可变参数的引用,当原数据发生修改时拷贝数据也会修改,考虑如下case:

>>> a = [1,2,[12]]
>>> b = a
>>> a = [2,3,[23]]
>>> a
[2, 3, [23]]
>>> b
[1, 2, [12]]

嘿?说好的同时修改呢?

实际上是因为a进行了一个赋值操作而非对象内的数据修改,因此指向了另外一片内存。浅拷贝的case如下:

>>> a = [1,2,[12]]
>>> b = a
>>> a[1] = 3
>>> a[2].append(3)
>>> b
[1, 3, [12, 3]]
>>>

对于序列来说,会发生浅拷贝的情况有切片,工厂函数list()和copy模块等。

深拷贝顾名思义是整个复制重开内存,这里不再赘述,不过有一点,python对于不可变变量,深拷也是直接拷引用的,反正不可变,节省空间~case如下:

>>> c = 1
>>> d = deepcopy(c)
>>> id(c),id(d)
(140569092332408, 140569092332408)

要特别注意元组类型,考虑如下case:

>>> a = (1,2)
>>> b = deepcopy(a)
>>> id(a),id(b)
(4482226008, 4482226008)
>>> a = (1,[12])
>>> b = deepcopy(a)
>>> id(a),id(b),id(a[0]),id(b[0]),id(a[1]),id(b[1])
(4482223208, 4482225792, 140569092332408, 140569092332408, 4482452008, 4482450280)

 

当元组数据均为immutable时,深拷是拷引用

当数据中有mutable时,深拷回开辟新内存,其中,immutable的变量指向同一处,mutable的指向不同处

另外,当元组中有可变数据进行浅拷时,可变数据修改也会同时发生。

————————————————————————————————————————————

immutable数据是可以自定义的,只要重写继承的object的__setattr__和__delattr__

 

  1. class Immutable(object):  
  2.     def __setattr__(self, *args):  
  3.         raise TypeError("can't modify the value of immutable instance")  
  4.   
  5.     __delattr__ = __setattr__  
  6.   
  7.     def __init__(self, value):  
  8.          super(Immutable, self).__setattr__("value", value)  

© 著作权归作者所有

共有 人打赏支持
守恒的猫
粉丝 4
博文 30
码字总数 16159
作品 0
房山
私信 提问
Python3开发随记(2)实例

<碎碎念> 今天在公司终于把jQuery搜索插件的功能写完了,已经能实现同步GET提交查询参数并正确显示视图,重构了三遍自己才觉得代码凑合能看,和高手还是有距离的,继续努力吧。 晚上继续看P...

hosealee
2016/04/13
98
0
Waston+Python,6行代码实现音频转文字,轻松做字幕~

几行代码,抱上IBM爸爸大腿,实现音频快速转文字~用到了IBM Waston的Speech to text(目前还是个demo) 碎碎念的废话~ 前几天工作中遇到需要把音频转为文字稿的任务,顺便学了个新词语:扒词...

莉莉安的向日葵
04/27
0
0
Python3开发随记(3)数据类型

<碎碎念> 上一个话题暂时不写了,直接写代码不好,还是先从基础开始吧 </碎碎念> Python3开发随记(3)数据类型 从这篇开始还是先看基础吧,直接看代码不好,另外我是一边看书一边写笔记的,...

hosealee
2016/04/17
72
0
svg、canvas、css3d实现数据可视化

前言: 这次项目用到了一些自定义的数据可视化组件,我把我做的部分抽出来几个典型做个汇总。 分为如下: 星球环绕旋转图 -- 方法一: svg:animateMotion+ animateTransform 方法二:css3d...

09/10
0
0
每日500字开篇

准备做一件有意义的事情——每日写500字的短文,我会尽量坚持下去,做不到日更,至少也要周更。 工作之余有不少空闲时间,大多数时候是在编码,最近在学 python,紧跟 AI 时代的潮流。人善于...

落英坠露
2017/12/11
0
0

没有更多内容

加载失败,请刷新页面

加载更多

可自定义扩展底部列表对话框ListBottomSheetDialogFragment

因为需要,为了方便,构建了一个可以自定义扩展的底部列表对话框,可以应付大部分场景。 效果图如下: 1.默认实现: 2.自定义列表实现 3.自定义头部和列表实现 一.可实现功能 1.默认可实现通...

明月春秋
58分钟前
1
0
数据库---增删改查

增:insert into 表名(列名1,列名2) values(‘列值1’,‘列值2’) 多行数据处理:insert into 表名(列名1,列名2) select ‘列值1’,‘列值2’ union select ‘列值1.1’,‘列值2.2...

森林之下
59分钟前
2
0
分布式/集群下session共享方案汇总

除去那些对容器依赖特别高的方案(如: 基于Tomcat的memcached-session-manager / tomcat-redis-session-manager,基于Jetty的jetty-nosql-memcache / jetty-session-redis ),自己整理了下...

哥本哈根的小哥
今天
2
0
Vue中的缩写:v-bind、v-on

v-bind 缩写:: 预期:any (with argument) | Object (without argument) 参数:attrOrProp (optional) 修饰符: .prop - 被用于绑定 DOM 属性。 .camel - (2.1.0+) 将 kebab-case 特性名转换......

文文1
今天
2
0
epoll中使用

1、一个线程epoll_wait时,另一个线程调用epoll_ctl是安全的。 2、使用edge触发,在socket有数据到来后,不收取数据,再次调用epoll_ctl将socket加入,仍会触发下一次动作。 asio用该方法来发...

gelare
今天
3
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部