一、前奏:熟悉Python内存管理

在Python中,变量在第一次赋值时自动声明,在创建—也就是赋值的时候,解释器会根据语法和右侧的操作数来决定新对象的类型。

引用计数器:一个内部跟踪变量

引用计数:每一个对象各有多少个引用

当对象被创建并(将其引用)赋值给变量时,该对象的引用计数就被设置为 1

x = 3.14

语句 x=3.14,创建一个浮点型对象并将其引用赋值给了x,x是第一个引用,该对象的引用计数为1

当一个对象(的引用)又被赋值到其他变量,或做参数传递等,该对象的一个新的引用(或叫别名)被创建,则该对象的引用计数自动+1。

以下都会增加引用计数:

y = x #做别名
foo(x) #做参数传递
mylis = [1,2,x,’a’] #成为容器对象的一个元素

以下都会减少引用计数:
复制代码

del x #del显式销毁

bar = x
x = True #对象的一个别名被赋值给其他对象

mylis.remove(x) #对象被从窗口对象中移除

del mylis #窗口对象本身被销毁

复制代码

二、Python的复制

从上面可见,对象的赋值实际上是对象的引用。当创建一个对象,然后把它赋给另一个变量的时候,python并没有拷贝这个对象,而只是拷贝了这个对象的引用。

当你对一个对象赋值的时候(做为参数传递,或者做为返回值),Python和Java一样,总是传递原始对象的引用,而不是一个副本。
复制代码

“”“传递原始对象的引用,而不是一个副本”“”
a = [1,2,3]
b = a
b.append(100)
print b #[1, 2, 3, 100]
print a #[1, 2, 3, 100]
print id(a) #11530368
print id(b) #11530368

复制代码

如 果你想修改一个对象,而且想让原始的对象不受影响,那你就需要对象复制。

可以 使用copy.copy(),它可以进行对象的浅复制(shallow copy),它复制了对象,但对于对象中的元素,依然使用引用.

(1)、使用切片[:]操作进行拷贝

(2)、使用工厂函数(如list/dir/set)等进行拷贝

(3)、copy.copy()
复制代码

jack = [‘jack’,[‘age’,20]]
tom = jack[:]
anny = list(jack)
jack
[‘jack’, [‘age’, 20]]
tom
[‘jack’, [‘age’, 20]]
anny
[‘jack’, [‘age’, 20]]
print id(jack),id(tom),id(anny)
13457088 18487376 18489136

复制代码

接下来修改上面例子,对姓名和年级进行修改:
复制代码

tom[0]=’tom’
anny[0]=’anny’
print tom
[‘tom’, [‘age’, 20]]
print anny
[‘anny’, [‘age’, 20]]
anny[1][1]
20
anny[1][1]= 18
anny[1][1]
18
print jack,tom,anny
[‘jack’, [‘age’, 18]] [‘tom’, [‘age’, 18]] [‘anny’, [‘age’, 18]]

复制代码

发现,虽然姓名都对号了,但是年龄却都变成了18.这是为什么呢?

我们看看它们元素的id
复制代码

[id(x) for x in jack]
[13463040, 13456608]
[id(x) for x in tom]
[13463424, 13456608]
[id(x) for x in anny]
[18501664, 13456608]

复制代码

发现,其中列表中 姓名字符串 id都不一样,但是 年龄列表id却都相同。

这是因为:python中字符串不可以修改,所以在为tom和anny重新命名的时候,会重新创建一个’tom’和’anny’对象,替换旧的’jack’对象。

这就说明了,浅复制(shallow copy),它复制了对象,但对于对象中的元素,依然使用引用.
同时还需搞清楚的是:python的list本身不包含对象。而是包含对象的引用。如:
li=[1,2,[A,B],5]
li[0] ,li[1] ,li[2]….他们只是一个变量名(或者说是标签)。它们之所以代表某个值,是因为他们引用了内存对象。这一点和C语言的数组比起来形成反差。

复制代码

“”“浅copy”“”
import copy
aa = [1,2,3]
bb = copy.copy(aa)
print id(aa) #11533088
print id(bb) #12014776
bb[0] =100
print bb #[100, 2, 3]
print aa #[1,2,3]

由于数字不可变,修改的时候会替换旧的对象

print [id(x) for x in bb] #[10247196, 10246388, 10246376]
print [id(y) for y in aa] #[10246400, 10246388, 10246376]

复制代码

下面试试对象中可变元素:
复制代码

lis = [[‘a’],[1,2],[‘z’,23]]
copyLis = copy.copy(lis)
copyLis[1].append(‘bar’)
print copyLis #[[‘a’], [1, 2, ‘bar’], [‘z’, 23]]
print lis #[[‘a’], [1, 2, ‘bar’], [‘z’, 23]]

复制代码

如果希望复制一个容器对象,以及它里面的所有元素(包含元素的子元素),使用copy.deepcopy,这个方法会消耗一些时间和空间,不过,如果你需要完全复制,这是唯一的方法.
复制代码

“”“深copy”“”
deepLis = copy.deepcopy(lis)
deepLis[1].append(‘foo’)
print deepLis #[[‘a’], [1, 2,’foo’], [‘z’, 23]]
print lis #[[‘a’], [1, 2], [‘z’, 23]]

复制代码

注意:

1、对于非容器类型(如数字、字符串、和其他‘原子’类型的对象)没有被拷贝一说。

2、如果元祖变量只包含原子类型对象,则不能深copy。

转自:http://www.cnblogs.com/BeginMan/p/3197649.html

Python的深拷贝与浅拷贝的更多相关文章

  1. python 中 深拷贝和浅拷贝的理解

    在总结 python 对象和引用的时候,想到其实 对于python的深拷贝和浅拷贝也可以很好对其的进行理解. 在python中,对象的赋值的其实就是对象的引用.也就是说,当创建一个对象,然后赋给另外一 ...

  2. [py]python的深拷贝和浅拷贝

    Python深复制浅复制or深拷贝浅拷贝 简单点说 copy.copy 浅拷贝 只拷贝父对象,不会拷贝对象的内部的子对象. copy.deepcopy 深拷贝 拷贝对象及其子对象 用一个简单的例子说明 ...

  3. Python中深拷贝与浅拷贝的区别

    转自:http://blog.csdn.net/u014745194/article/details/70271868 定义: 在Python中对象的赋值其实就是对象的引用.当创建一个对象,把它赋值给 ...

  4. python中深拷贝与浅拷贝

    # 1.浅拷贝(复制东西)a = [11,22,33] # 实际上是浅拷贝# 没有把这个变量的值赋进去,而是把另一个变量的地址拿过去了,就叫浅拷贝.b = a # print(id(a))# prin ...

  5. python之深拷贝和浅拷贝

    1.当拷贝的是不可变数据类型(数值.字符串.元组),不管是深拷贝和浅拷贝,都指向的是同一地址: 2.当拷贝的对象是可变数据类型(列表.字典): (1)当浅拷贝的对象中无复杂子对象,原来值的改变不会影响 ...

  6. python中深拷贝和浅拷贝

    python中所谓浅拷贝就是对引用的拷贝,所谓深拷贝就是对对象的资源的拷贝. 首先,对赋值操作我们要有以下认识: 赋值是将一个对象的地址赋值给一个变量,让变量指向该地址( 旧瓶装旧酒 ). 修改不可变 ...

  7. python 中深拷贝和浅拷贝的区别

    通俗的理解,浅就是外面,深就是里面.浅拷贝的意思就是只拷贝外面的一层,深拷贝就是拷贝的里面的所有. 看两段代码: 元组: #!/usr/bin/env/python # -*-coding:utf-8 ...

  8. 关于Python中深拷贝与浅拷贝的理解(一)---概念

    import copy a = [1, 2, 3, 4, ['a', 'b']] #原始对象 b = a #赋值,传对象的引用 c = copy.copy(a) #对象拷贝,浅拷贝 d = copy. ...

  9. Python中深拷贝与浅拷贝区别

    浅拷贝, list值是可变的,str值不可变,只能重新赋值 a=b=c='wjx'print(a,b,c)c= 'jmy'#重新赋值了,所以内存分配了新的地址print(a,b,c)print(id( ...

  10. Python的深拷贝、浅拷贝

    浅拷贝 定义:浅拷贝只是对另外一个变量的内存地址的拷贝,这两个变量指向同一个内存地址的变量值. 浅拷贝的特点: 公用一个值: 这两个变量的内存地址一样: 对其中一个变量的值改变,另外一个变量的值也会改 ...

随机推荐

  1. poj 2524 并查集 Ubiquitous Religions

    //#include<bits/stdc++.h> #include<iostream> #include<stdio.h> #define max1 50005 ...

  2. Playmaker Input篇教程之PlayMaker菜单概述

    Playmaker Input篇教程之PlayMaker菜单概述 Playmaker InputPlayMaker菜单概述 Playmaker插件被导入游戏项目以后,会自动为Unity编辑器添加一个名 ...

  3. 错误3 error C3859: 超过了 PCH 的虚拟内存范围;请使用“-Zm120”

    在工程上单击右键,属性,C/C++,最后一项,命令行,在里面/Zm200或者编译器提示的大小解决.

  4. 浅谈Apache Spark的6个发光点(CSDN)

    Spark是一个基于内存计算的开源集群计算系统,目的是更快速的进行数据分析.Spark由加州伯克利大学AMP实验室Matei为主的小团队使用Scala开发开发,其核心部分的代码只有63个Scala文件 ...

  5. 十个JavaScript中易犯的小错误,你中了几枪?

    序言 在今天,JavaScript已经成为了网页编辑的核心.尤其是过去的几年,互联网见证了在SPA开发.图形处理.交互等方面大量JS库的出现. 如果初次打交道,很多人会觉得js很简单.确实,对于很多有 ...

  6. BZOJ4011: [HNOI2015]落忆枫音

    Description 「恒逸,你相信灵魂的存在吗?」 郭恒逸和姚枫茜漫步在枫音乡的街道上.望着漫天飞舞的红枫,枫茜突然问出 这样一个问题.  「相信吧.不然我们是什么,一团肉吗?要不是有灵魂……我们 ...

  7. Hibernate工作原理及为什么要用?

    Hibernate工作原理及为什么要用? 原理:1.通过Configuration().configure();读取并解析hibernate.cfg.xml配置文件2.由hibernate.cfg.x ...

  8. com.mysql.jdbc.exceptions.jdbc4.MySQLSyntaxErrorException: Unknown column '??????' in 'field list'

    严重: Servlet.service() for servlet jsp threw exceptioncom.mysql.jdbc.exceptions.jdbc4.MySQLSyntaxErro ...

  9. for循环下九九乘法表

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...

  10. 将java的class文件放到一个指定文件夹下

    用javac执行java文件时,要把java文件的class文件放到指定文件夹下,注意文件夹要创建好,执行javac -d 文件夹 ***.java 如图: 在class文件夹下就出现了L的class ...