Python开发【第二章】:深浅拷贝剖析
Python深浅拷贝剖析
Python中,对象的赋值,拷贝(深/浅拷贝)之间是有差异的,如果使用的时候不注意,就可能产生意外的结果。
下面本文就通过简单的例子介绍一下这些概念之间的差别。
一、对象赋值
创建列表变量Alex,变量包含子列表,通过变量Alex给变量lzl赋值,然后对变量Alex的元素进行修改,此时lzl会有什么变化呢?让我们通过内存地址分析两者的变化
# 对象赋值
import copy #import调用copy模块 Alex = ["Alex", 28, ["Python", "C#", "JavaScript"]]
lzl = Alex #直接赋值 # 修改前打印
print(id(Alex))
print(Alex)
print([id(adr) for adr in Alex])
# 输出: 7316664
# ['Alex', 28, ['Python', 'C#', 'JavaScript']]
# [2775776, 1398430400, 7318024]
print(id(lzl))
print(lzl)
print([id(adr) for adr in lzl])
# 输出: 7316664
# ['Alex', 28, ['Python', 'C#', 'JavaScript']]
# [2775776, 1398430400, 7318024] # 对变量进行修改
Alex[0]='Mr.Wu'
Alex[2].append('CSS')
print(id(Alex))
print(Alex)
print([id(adr) for adr in Alex])
# 输出: 7316664
# ['Mr.Wu', 28, ['Python', 'C#', 'JavaScript', 'CSS']]
# [5170528, 1398430400, 7318024]
print(id(lzl))
print(lzl)
print([id(adr) for adr in lzl])
# 输出: 7316664
# ['Mr.Wu', 28, ['Python', 'C#', 'JavaScript', 'CSS']]
# [5170528, 1398430400, 7318024]
通过上面的代码做出如下两图并进行分析:
1、首先,创建了一个名为Alex的变量,这个变量指向一个list列表,从第一张图中可以看到列表中元素的地址(每次运行,结果可能不同)。然后通过变量Alex给变量lzl进行赋值,变量lzl指向Alex指向的内存地址(7316664),所有可以理解为,Python中,对象的赋值都是进行对象引用(内存地址)的传递,被赋值的变量并没有开辟新内存,两个变量共用一个内存地址
2、第二张图中,由于Alex和lzl指向同一个对象(内存地址),所以对Alex的任何修改都会体现在lzl上。这里需要注意的一点是,str是不可变类型,所以当修改元素Alex为Mr.Wu时,内存地址由2775776变为了5170528,list是可变类型,元素['Python', 'C#', 'JavaScript', 'CSS']修改完后,内存地址仍然是7318024,没有发生改变

---------------------------------------------------------------------------------------

二、浅拷贝
创建列表变量Alex,变量包含子列表,通过copy模块的浅拷贝函数copy()对变量Alex进行拷贝,当对Alex进行操作时,此时lzl会如何变化?
# 浅拷贝
import copy #import调用copy模块 Alex = ["Alex", 28, ["Python", "C#", "JavaScript"]]
lzl = copy.copy(Alex) #通过copy模块里面的浅拷贝函数copy() # 修改前打印
print(id(Alex))
print(Alex)
print([id(adr) for adr in Alex])
# 输出: 10462472
# ['Alex', 28, ['Python', 'C#', 'JavaScript']]
# [5462752, 1359960768, 10463232]
print(id(lzl))
print(lzl)
print([id(adr) for adr in lzl])
# 输出: 10201848
# ['Alex', 28, ['Python', 'C#', 'JavaScript']]
# [5462752, 1359960768, 10463232] # 对变量进行修改
Alex[0]='Mr.Wu'
Alex[2].append('CSS')
print(id(Alex))
print(Alex)
print([id(adr) for adr in Alex])
# 输出: 10462472
# ['Mr.Wu', 28, ['Python', 'C#', 'JavaScript', 'CSS']]
# [10151264, 1359960768, 10463232]
print(id(lzl))
print(lzl)
print([id(adr) for adr in lzl])
# 输出: 10201848
# ['Alex', 28, ['Python', 'C#', 'JavaScript', 'CSS']]
# [5462752, 1359960768, 10463232]
通过上面的代码做出如下两图并进行分析:
1、依然使用一个Alex变量,指向一个list类型的对象,list包含一个子list
2、然后,通过copy模块里面的浅拷贝函数copy(),对Alex指向的对象进行浅拷贝,然后浅拷贝生成的新对象赋值给lzl变量,注意此时变量lzl新建了一块内存(10201848),此内存记录了list中元素的地址,对于list中的元素,浅拷贝就会使用原始元素的引用(内存地址)
3、当对Alex进行修改的时候,由于list中第一个元素“Alex”(str)为不可变类型,所以进行修改的后,第一个元素对应的地址变为了10151264。由于list中第三个元素['Python', 'C#', 'JavaScript'](list)为可变类型,修改后地址没有变化。所以最后变量lzl和变量Alex只是第一个元素不一样。

-----------------------------------------------------------------------------------------

注:当我们使用下面的操作的时候,会产生浅拷贝的效果:
- 使用切片[:]操作
- 使用工厂函数(如list/dir/set)
- 使用copy模块中的copy()函数
三、深拷贝
创建列表变量Alex,变量包含子列表,通过copy模块的深拷贝函数deepcopy()对变量Alex进行拷贝,当对Alex进行操作时,此时lzl会如何变化?
#!/usr/bin/env python
# -*- coding:utf-8 -*-
#-Author-Lian
# 深拷贝
import copy #import调用copy模块 Alex = ["Alex", 28, ["Python", "C#", "JavaScript"]]
lzl = copy.deepcopy(Alex) #通过copy模块里面的深拷贝函数deepcopy() # 修改前打印
print(id(Alex))
print(Alex)
print([id(adr) for adr in Alex])
# 输出: 6202712
# ['Alex', 28, ['Python', 'C#', 'JavaScript']]
# [4086496, 1363237568, 6203472]
print(id(lzl))
print(lzl)
print([id(adr) for adr in lzl])
# 输出: 6203032
# ['Alex', 28, ['Python', 'C#', 'JavaScript']]
# [4086496, 1363237568, 6203512] # 对变量进行修改
Alex[0]='Mr.Wu'
Alex[2].append('CSS')
print(id(Alex))
print(Alex)
print([id(adr) for adr in Alex])
# 输出: 6202712
# ['Mr.Wu', 28, ['Python', 'C#', 'JavaScript', 'CSS']]
# [5236064, 1363237568, 6203472]
print(id(lzl))
print(lzl)
print([id(adr) for adr in lzl])
# 输出: 6203032
# ['Alex', 28, ['Python', 'C#', 'JavaScript']]
# [4086496, 1363237568, 6203512]
通过上面深拷贝的代码做出如下两图并进行分析:
1、依然使用一个Alex变量,指向一个list类型的对象,list包含一个子list
2、然后,通过copy模块里面的深拷贝函数deepcopy(),对Alex指向的对象进行深拷贝,然后深拷贝生成的新对象赋值给lzl变量。跟浅拷贝一样,此时变量lzl依然新建了一块内存(6203032),此内存记录了list中元素的地址。但是,对于list中的元素,深拷贝不是简单的使用原始元素的引用(内存地址),对于list第三个元素(['Python', 'C#', 'JavaScript'])重新生成了一个地址(6203512),此时两个变量的第三个元素的内存引用地址不同
3、当对Alex进行修改的时候,由于list中第一个元素“Alex”(str)为不可变类型,所以进行修改的后,第一个元素对应的地址变为了5236064。虽然list中第三个元素['Python', 'C#', 'JavaScript'](list)为可变类型,修改后不会产生新的地址,但是由于Alex和lzl在第三个元素引用的本就不同,所有Alex的修改对lzl不会产生任何影响

-------------------------------------------------------------------------------------

其实,对于拷贝有一些特殊情况:
- 对于非容器类型(如数字、字符串、和其他'原子'类型的对象)没有拷贝这一说
- 也就是说,对于这些类型,"obj is copy.copy(obj)" 、"obj is copy.deepcopy(obj)"
- 如果元祖变量只包含原子类型对象,则不能深拷贝
Python开发【第二章】:深浅拷贝剖析的更多相关文章
- 路飞学城-Python开发-第二章
''' 数据结构: menu = { '北京':{ '海淀':{ '五道口':{ 'soho':{}, '网易':{}, 'google':{} }, '中关村':{ '爱奇艺':{}, '汽车之家' ...
- Python开发——数据结构【深浅拷贝】
浅拷贝 # 浅拷贝只copy一层 s = [3,'Lucy',4,[1,2]] s1 = s.copy() 深拷贝 # 深拷贝——克隆一分 import copy s = [3,'Lucy',4,[1 ...
- Python开发【第二章】:Python深浅拷贝剖析
Python深浅拷贝剖析 Python中,对象的赋值,拷贝(深/浅拷贝)之间是有差异的,如果使用的时候不注意,就可能产生意外的结果. 下面本文就通过简单的例子介绍一下这些概念之间的差别. 一.对象赋值 ...
- ASP.NET自定义控件组件开发 第二章 继承WebControl的自定义控件
原文:ASP.NET自定义控件组件开发 第二章 继承WebControl的自定义控件 第二章 继承于WebControl的自定义控件 到现在为止,我已经写了三篇关于自定义控件开发的文章,很感谢大家的支 ...
- python直接赋值、深浅拷贝实例剖析
根据数据类型分为两部分进行剖析: int.str类型 list.tuple.dict类型等 1. int.str类型 [int类型实例] >>> import copy ...
- python测试开发面试之深浅拷贝
先来道题热热身 a = ('a', 'b','c') c = copy.copy(a) d = copy.deepcopy(a) if c == d: print("c和d的值相等" ...
- python之路(三)-深浅拷贝
深浅拷贝用法来自copy模块. 导入模块:import copy 浅拷贝:copy.copy 深拷贝:deepcopy 字面理解:浅拷贝指仅仅拷贝数据集合的第一层数据,深拷贝指拷贝数据集合的所有层.所 ...
- Android 系统移植与驱动开发--第二章搭建Android环境核心步骤及心得
第二章 搭建Android 开发环境 虽然在这一章中讲的是Android底层开发环境,但是相应伴随的还有Android NDK程序来测试Linux驱动,HAL程序库.底层开发不仅需要交叉编译环境,还要 ...
- [Python笔记][第二章Python序列-复杂的数据结构]
2016/1/27学习内容 第二章 Python序列-复杂的数据结构 堆 import heapq #添加元素进堆 heapq.heappush(heap,n) #小根堆堆顶 heapq.heappo ...
随机推荐
- vue组件的原理
https://www.cnblogs.com/landeanfen/p/6518679.html
- C++生成DM数据点导入DM
在c++编写正弦曲线点的代码,源代码如下: //想要使用内置的π,此句必不可少! #define _USE_MATH_DEFINES #include<iostream> #include ...
- LeetCode 第 149 场周赛
成绩 一.一年中的第几天(LeetCode-1154) 1.1 题目描述 1.2 解题思路 比较容易的一题,搞清楚平年.闰年的判定规则,就很容易做出来. 1.3 解题代码 class Solution ...
- 5分钟学会如何创建spring boot项目
上一篇博客说了如何创建spring boot项目,但是有些同学会觉得有点麻烦,有没有什么快速学会能快速创建spring boot项目的方法,答案是肯定的.接下来我们就一起来快速创建一个spring b ...
- Go 语言入门(三)并发
写在前面 在学习 Go 语言之前,我自己是有一定的 Java 和 C++ 基础的,这篇文章主要是基于A tour of Go编写的,主要是希望记录一下自己的学习历程,加深自己的理解 Go 语言入门(三 ...
- Linux中 mv(文件移动)
mv命令的功能有以下两种: source target mv 参数1 参数2 1.对文件或目录重新命名 如果源文件和目标文件在同一个目录下,mv的作用就是改文件名. 2.将文件从一个目录移到另一个目录 ...
- Python自学笔记(九)
#类 #类的创建 :class类名 + 冒号,后面语句要缩进 #类的属性创建:通过赋值语句(即定义“是怎样的”) #实例方法的创建:def + 方法名(self) #方法具体的执行过程,即定义“能做什 ...
- linux几种传输方式与拷贝方式的性能分析
本文记录linux系统中文件传输的多种方式,留作备忘.linux中文件传输的方式有ftp,scp,rsync,rz,sz等,但各个工具的功能又有所区别: FTP : FTP是文件服务器,可实现文件的上 ...
- 猎豹网校C++ Primer学习笔记
1.头文件(15th课) 大型项目开发,要有很多头文件.只能写声明,不能定义(类定义和常量定义可以). 自己新建头文件(类定义,外部变量声明,函数声明).源文件包含对应的头文件. 头文件里写类的声明, ...
- 机器学习 - 算法 - Xgboost 数学原理推导
工作原理 基于集成算法的多个树累加, 可以理解为是弱分类器的提升模型 公式表达 基本公式 目标函数 目标函数这里加入了损失函数计算 这里的公式是用的均方误差方式来计算 最优函数解 要对所有的样本的损失 ...