对象赋值实际上是简单的对象引用。也就是说当你创建一个对象,然后把它赋给另一个变量的时候,Python并没有拷贝这个对象,而只是拷贝了这个对象的引用。

假设想创建一对小夫妻的通用档案,名为person。然后你分别为他俩拷贝一份。

在下面的例子中,我们展示了两种拷贝对象的方式,一种使用了切片操作,另一种用了工厂方法,为了区分出三个不同的对象,我们使用id()内建函数来显示每个对象的标识符(还可以用is 操作符来做相同的事情)。第一步:

>>> person = ['name', ['savings',100.00]]

>>> hubby = person[:]                                   # slice copy

>>> wifey = list(person)       
                        # fac func copy

>>> [id(x) for x in person,hubby, wifey]

[23554728, ,
]          #三个不同的对象

现在为他们创建了初始有$100 的个人存款帐户。接下来将用户名改为定制的名字。并且让丈夫取走$50。第二步代码如下:

>>> hubby[0] = 'joe'

>>> wifey[0] = 'jane'

>>> hubby, wifey

(['joe', ['savings', 100.0]], ['jane',['savings', 100.0]])

>>> hubby[1][1] = 50.00

>>> hubby, wifey

(['joe', ['savings', 50.0]], ['jane', ['savings', 50.0]])

可见丈夫的行为影响到了妻子的账户,虽然我们进行了分开的拷贝作。为什么会这样呢?

原因是我们仅仅做了一个浅拷贝。对一个对象进行浅拷贝其实是新创建了一个类型跟原对象一样,其内容是原来对象元素的引用,换句话说,这个拷贝的对象本身是新的,但是它的内容不是。

在第一步进行完初始化之后,执行下列代码的结果如下:

>>> [id(x) for x in person]

[,
]

>>> [id(x) for x in hubby]

[,
]

>>> [id(x) for x in wifey]

[,
]

可见,三个列表虽然有各自的ID,但是列表中的元素的ID却是相同的。这也就是浅拷贝的意思。所以当执行hubby[1][1]
=50.00之后,它改变了三个列表中,共同指向的内部列表。因此这句代码影响到了三个列表。

序列类型对象的浅拷贝是默认类型拷贝,并可以以下几种方式实施:

(1)完全切片操作[:]

(2)利用工厂函数,比如list(),dict()等

(3)使用copy模块的copy函数.

你的下一个问题可能是:当妻子的名字被赋值,为什么丈夫的名字没有受到影响?难道它们的名字现在不应该都是'jane'了吗?为什么名字没有变成一样的呢?

这是因为在这两个列表的两个对象中,第一个对象是不可变的(是个字符串类型),而第二个是可变的(一个列表)。正因如此,当修改妻子的名字时,是新创建了一个字符串对象。

作完第二步之后,让我们分别看一下每个列表的元素的对象ID值。

>>> [id(x) for x in person]

[, 23575208]

>>> [id(x) for x in hubby]

[, 23575208]

>>> [id(x) for x in wifey]

[, 23575208]

可见,三个列表的第一个元素各不相同,但是第二个元素确实相同的引用。

如果需要的是两个分离账户,就需要进行深拷贝了.深拷贝----创建一个新的容器对象,包含原有对象元素(引用)全新拷贝的引用。这需要copy.deepcopy()函数.我们使用深拷贝来重写整个例子.

>>> person = ['name', ['savings',100.00]]

>>> hubby = person

>>> import copy

>>> wifey = copy.deepcopy(person)

>>> [id(x) for x in person,hubby, wifey]

[12242056, ,
]

>>> hubby[0] = 'joe'

>>> wifey[0] = 'jane'

>>> hubby, wifey

(['joe', ['savings', 100.0]], ['jane',['savings', 100.0]])

>>> hubby[1][1] = 50.00

>>> hubby, wifey

(['joe', ['savings', 50.0]], ['jane',['savings', 100.0]])

这就是我们想要的方式,作为验证,让我们确认一下所有四个对象都是不同的.

>>> [id(x) for x in hubby]

[12191712, 11826280]

>>> [id(x) for x in wifey]

[12114080, 12224792]

以下有几点关于拷贝操作的警告。

        第一:非容器类型(比如数字,字符串和其他"原子"类型的对象,像代码、类型和xrange对象等)没有拷贝一说。

第二:如果是元组类型,则切片拷贝和工厂函数拷贝,只是增加引用计数而已。如果使用copy.deepcopy函数,则会创建新的元组类型,但是还是浅拷贝。

>>> person = ('name', ['savings',100.00])

>>> hubby = person[:]

>>> wifey = tuple(person)

>>>

>>>

>>> [id(x) for x in person,hubby, wifey]

[23575888, 23575888,23575888]

>>> [id(x) for x in person]

[18859168, 23574848]

>>> [id(x) for x in hubby]

[18859168, 23574848]

>>> [id(x) for x in wifey]

[18859168, 23574848]

>>> wifey = copy.deepcopy(person)

>>> [id(x) for x in person,hubby, wifey]

[23575888, 23575888, ]

>>> [id(x) for x in wifey]

[18859168, 23554648]

>>> person = ['name', ('savings',100.00)]

>>> newPerson =copy.deepcopy(person)

>>> [id(x) for x in person,newPerson]

[12225352, 12226112]

>>> [id(x) for x in person]

[,
]

>>> [id(x) for x in newPerson]

[,
]

浅拷贝和深拷贝操作都可以在copy模块中找到。其实copy模块中只有两个函数可用:copy()进行浅拷贝操作,而deepcopy()进行深拷贝操作。

参考:《python核心编程》

Python深入:02浅拷贝深拷贝的更多相关文章

  1. python开发_copy(浅拷贝|深拷贝)_博主推荐

    在python中,有着深拷贝和浅拷贝,即copy模块 下面我们就来聊一下: 运行效果: ================================================== 代码部分: ...

  2. Python中的浅拷贝 深拷贝

    浅拷贝只拷贝父对象,子对象的地址空间不改变,包括下面三种: 1. copy 从下面的例子可以看出对象c从a拷贝,当对象a增加一个列表元素之后,c对象没有改变, 而当对象a中的子列表改变时,对象c的子列 ...

  3. PythonStudy1——Python 值拷贝 浅拷贝 深拷贝

    拷贝:对值进行复制的过程 # 值拷贝:应用场景最多  ls = [1, 'abc', [10]] ls1 = ls # ls1直接将ls中存放的地址拿过来  # ls内部的值发生任何变化,ls1都会随 ...

  4. python中赋值-浅拷贝-深拷贝之间的关系

    赋值: 变量的引用,没有拷贝空间 对象之间赋值本质上 是对象之间的引用传递而已.也就是多个对象指向同一个数据空间. 拷贝的对象分两种类型: . 拷贝可变类型 浅拷贝: 只拷贝第一层数据,不关心里面的第 ...

  5. Python练习四-浅拷贝&深拷贝

    一.数字.字符串不论是浅拷贝.深拷贝都是指向一个地址. a = 1 b = "abc" print (id(a)) print (id(b)) a1 = a b1 = b prin ...

  6. Python学习02 列表 List

    Python学习02 列表 List Python列表 List Python中的列表(List)用逗号分隔,方括号包围(comma-separated values (items) between ...

  7. Python网络02 Python服务器进化

    原文:Python网络02 Python服务器进化 作者:Vamei 出处:http://www.cnblogs.com/vamei 欢迎转载,也请保留这段声明.谢谢! **注意,在Python 3. ...

  8. JS中有关对象的继承以及实例化、浅拷贝深拷贝的奥秘

    一.属性的归属问题 JS对象中定义的属性和方法如果不是挂在原型链上的方法和属性(直接通过如类似x的方式进行定义)都只是在该对象上,对原型链上的没有影响.对于所有实例共用的方法可直接定义在原型链上这样实 ...

  9. $.extend()浅拷贝深拷贝

    参考网址:http://bijian1013.iteye.com/blog/2255037 jQuery.extend() 函数用于将一个或多个对象的内容合并到目标对象. 注意:1. 如果只为$.ex ...

随机推荐

  1. Shell 常用特性

       管道(|) 管道 (|): 将一个命令的输出作为另外一个命令的输入.   管道同样可以在标准输入输出和标准错误输出间做代替工作,这样一来,可以将某一个程序的输出送到另一个程序的输入,其语法如下: ...

  2. Java问题解读系列之String相关---String类为什么是final的?

    今天看到一篇名为<Java开发岗位面试题归类汇总>的博客,戳进去看了一下题目,觉得有必要夯实一下基本功了,所以打算边学边以博客的形式归纳总结,每天一道题, 并将该计划称为java问题解读系 ...

  3. Spring Boot Redis 集成配置(转)

    Spring Boot Redis 集成配置 .embody{ padding:10px 10px 10px; margin:0 -20px; border-bottom:solid 1px #ede ...

  4. Leetcode48. Rotate Image旋转图像

    给定一个 n × n 的二维矩阵表示一个图像. 将图像顺时针旋转 90 度. 说明: 你必须在原地旋转图像,这意味着你需要直接修改输入的二维矩阵.请不要使用另一个矩阵来旋转图像. 示例 1: 给定 m ...

  5. vue源码解读-目录结构

    目录结构 ├── scripts ------------------------------- 构建相关的文件,一般情况下我们不需要动│ ├── git-hooks ---------------- ...

  6. tomcat 端口占用问题解决

    在dos下,输入  netstat   -ano|findstr  8080 //说明:查看占用8080端口的进程 显示占用端口的进程 taskkill  /pid  6856  /f //说明,运行 ...

  7. Sentinel 1.5.0 正式发布,引入 Reactive 支持

    近日,流控降级组件 Sentinel 的又一个里程碑版本 1.5.0 正式发布. 该版本引入 Reactive 的支持,并提供多项新特性与改进.从 1.5.0 版本开始,Sentinel 仅支持 JD ...

  8. 自定义View系列教程01--常用工具介绍

    站在源码的肩膀上全解Scroller工作机制 Android多分辨率适配框架(1)- 核心基础 Android多分辨率适配框架(2)- 原理剖析 Android多分辨率适配框架(3)- 使用指南 自定 ...

  9. SQLServer —— 视图

    一.视图的概念 是存储在服务器端的一个查询块,是一张虚拟表. 表示一张表的部分数据或多张表的综合数据. 其结构和数据是建立在对表的查询基础上. 视图的使用,跟对普通的表的查询使用完全一样. 二.视图中 ...

  10. mysqldump命令之数据库迁移

    格式说明如下: mysqldump -h源主机IP -u源主机用户 -p源主机用户密码 数据库名 | mysql -h目标主机IP -u目标主机用户 -p目标用户密码 数据库名