『Python基础』第20节:深浅copy
一. 赋值运算
l1 = [1, 2, 'conan', [11, 22]]
l2 = l1
l1[0] = 111
print(l1) # [111, 2, 'conan', [11, 22]]
print(l2) # [111, 2, 'conan', [11, 22]]
print(id(l1[0])) # 1872924208
print(id(l2[0])) # 1872924208
l1.append(333)
print(l1) # [111, 2, 'conan', [11, 22], 333]
print(l2) # [111, 2, 'conan', [11, 22], 333]
print(id(l1)) # 2211423993544
print(id(l2)) # 2211423993544
l1[3].append(666)
print(l1) # [111, 2, 'conan', [11, 22, 666], 333]
print(l2) # [111, 2, 'conan', [11, 22, 666], 333]
print(id(l1[3])) # 2211453800136
print(id(l2[3])) # 2211453800136
从上面的运行结果可以看到, 对于赋值运算来说, l1与l2指向的是同一个内存地址, 所以它们是完全一样的. 其中一个变量对列表进行改变, 剩下的变量在使用列表时, 就是使用的改变之后的列表.

二. 浅copy
# 同一代码块下
l1 = [1, 2, 'conan', [11, 22]]
l2 = l1.copy()
l1[0] = 111
print(l1) # [111, 2, 'conan', [11, 22]]
print(l2) # [1, 2, 'conan', [11, 22]]
print(id(l1[0])) # 1872924208
print(id(l2[0])) # 1872920688
print(id(l1[1])) # 1872920720
print(id(l2[1])) # 1872920720
l1.append(333)
print(l1) # [111, 2, 'conan', [11, 22], 333]
print(l2) # [1, 2, 'conan', [11, 22]]
print(id(l1)) # 2343138115272
print(id(l2)) # 2343138954824
l1[3].append(666)
print(l1) # [111, 2, 'conan', [11, 22, 666], 333]
print(l2) # [1, 2, 'conan', [11, 22, 666]]
print(id(l1[3])) # 2343138954952
print(id(l2[3])) # 2343138954952
# 不同代码块下
l1 = [1, 2, 'conan', [11, 22]]
l2 = l1.copy()
l1[0] = 111
print(l1) # [111, 2, 'conan', [11, 22]]
print(l2) # [1, 2, 'conan', [11, 22]]
print(id(l1[0])) # 1872924208
print(id(l2[0])) # 1872920688
print(id(l1[1])) # 1872920720
print(id(l2[1])) # 1872920720
l1.append(333)
print(l1) # [111, 2, 'conan', [11, 22], 333]
print(l2) # [1, 2, 'conan', [11, 22]]
print(id(l1)) # 2402040017736
print(id(l2)) # 2402040018120
l1[3].append(666)
print(l1) # [111, 2, 'conan', [11, 22, 666], 333]
print(l2) # [1, 2, 'conan', [11, 22, 666]]
print(id(l1[3])) # 2402040003848
print(id(l2[3])) # 2402040003848
对于浅copy来说, 只是在内存中重新开辟了一个空间存放一个新列表, 但是新列表里面的元素与原列表里面的元素的内存地址是同一个. 所以, l1与l2的id不同, 但是里面内容的id是相同的.
注意
我们发现, 在更改l1的第一个元素, 也就是把 1 改为 111 之后, 内存地址是不一样的, 这是因为int类型是不可变的数据类型; 而列表时可变的数据类型, 当里面的列表增加一个元素后, 其内存地址是不变的.

三. 深copy
依旧是先看下代码
import copy
l1 = [1, 2, 'conan', [11, 22]]
l2 = copy.deepcopy(l1)
l1[0] = 111
print(l1) # [111, 2, 'conan', [11, 22]]
print(l2) # [1, 2, 'conan', [11, 22]]
print(id(l1[0])) # 1872924208
print(id(l2[0])) # 1872920688
print(id(l1[1])) # 1872920720
print(id(l2[1])) # 1872920720
l1.append(333)
print(l1) # [111, 2, 'conan', [11, 22], 333]
print(l2) # [1, 2, 'conan', [11, 22]]
print(id(l1)) # 2698658605000
print(id(l2)) # 2698660016776
l1[3].append(666)
print(l1) # [111, 2, 'conan', [11, 22, 666], 333]
print(l2) # [1, 2, 'conan', [11, 22]]
print(id(l1[3])) # 2698659950152
print(id(l2[3])) # 2698660016712
对于深copy来说, 深copy后的列表是在内存中重新创建的. 而且Python为了节省内存以及提高性能, 它做了这么一件事:
在深copy后的列表中,
如果其元素是可变的数据类型就再重新创建一个;
如果其元素是不可变的数据类型, 就公用一个, 也就是其内存地址是一样的.
在上面代码中的运行结果也能看出来.

四. 总结
浅copy: 嵌套的可变的数据类型 是 同一个.
深copy: 嵌套的可变的数据类型 不是 同一个.
五. 练习
5.1 练习1
写出下面代码的运行结果
l1 = [1, 2, 3, ['conan']]
l2 = l1[::]
l1[-1].append(666)
print(l1)
print(l2)
l1: [1, 2, 3, ['conan', 666]]
l2: [1, 2, 3, ['conan', 666]]
通过打印其id可以发现, l2 = l1[::] 的操作其实与浅copy一样.
『Python基础』第20节:深浅copy的更多相关文章
- 『Python基础』第2节: Python简介及入门
一. Python介绍 Python是一门高级计算机程序设计语言,1989年,荷兰的Guido von Rossum创造了它.Guido是是一个牛人,1982年,他从阿姆斯特丹大学获得了数学和计算机硕 ...
- 『Python基础』第4节:基础数据类型初识
本节只是对基础数据类型做个简单介绍, 详情会在之后慢慢介绍 什么是数据类型? 我们人类可以分清数字与字符串的区别, 可是计算机不能. 虽然计算机很强大, 但在某种程度上又很傻, 除非你明确告诉它数字与 ...
- 『Python基础』第7节:基本运算符
一. 基本运算符 运算按种类可以分为: 算数运算.比较运算.逻辑运算.赋值运算.成员运算.身份运算.位运算. 今天我们只学习算数运算.比较运算.逻辑运算.赋值运算.成员运算 1.1 算数运算 以下假设 ...
- 『Python基础』第5节:条件控制
if 语句的使用 单分支 if 条件: 满足条件后要执行的代码 例如: if 2 < 3: print(222) print(333) 每个条件后面都要使用冒号 :, 表示接下来是满足条件后要执 ...
- 『Python基础』第39节 函数的返回值
1. 函数的返回值 一个函数就是封装一个功能, 这个功能一般都会有一个最终结果的. 比如写一个登录的函数, 最终登录是否成功你总得告诉我一声吧? 还有咱们之前也用过 len() 这个函数, ...
- 『Python基础』第8节:格式化输出
现在有一个需求, 询问用户的姓名, 年龄, 工作, 爱好, 然后打印成以下格式 ************ info of Conan ************ name: Conan age: 23 ...
- 『Python基础』第6节:流程控制之while循环
在生活中经常遇到循环的事情, 比如循环列表播放歌曲等. 在Python中, 也有循环, 就是其流程控制语句while. 1. 基本循环 while 条件: 循环体 # 如果条件为真, 那么就执行循环体 ...
- 『Python基础』第1节 Windows环境下安装Python3.x
一. Python安装 1. 下载安装包 https://www.python.org/downloads/release/python-374/ # 3.7安装包 # 如需安装python2.7版本 ...
- 『Go基础』第8节 格式化输出
输出就是将数据信息打印到电脑屏幕上. 本节我们就来学习一下Go语言中的三种输出方式: Print().Println().Printf(). 1.Print() Print()主要的一个特点就是打印数 ...
随机推荐
- sass登陆页面实例
sass登陆页面实例 一.总结 一句话总结: sass使用非常方便:使用就是将sass转化为css引入,并且动态监听让sass转化为css,可以很方便的所见即所得 1.sass安装? npm就可以按照 ...
- iview3 版本 升级
Button 废弃 type ghost,原先的 default 样式有改变. Icon 的图标升级至 ionicons 3.0 图标,图标名称有改变. Breadcrumb 废弃 href 属性. ...
- kubernetes 亲和性调度详解
文章目录 1 概述: 2 场景一:调度到一组具有相同特性的主机上(label+nodeSelector) 3 场景二:部署的应用不想调度到某些节点上(nodeaffinity) 4 场景三:部署的应用 ...
- openresty开发系列2--nginx的简单安装,正向、反向代理及常用命令和信号控制介绍
openresty开发系列2--nginx的简单安装,正向.反向代理及常用命令和信号控制介绍 一.nginx的安装下载及编译安装1.Nginx下载:nginx-1.13.0.tar.gz,下载到:/u ...
- shell编程系列14--文本处理三剑客之awk的概述及常用方法总结
shell编程系列14--文本处理三剑客之awk的概述及常用方法总结 awk是一个文本处理工具,通常用于处理数据并生成结果报告 awk的命名是它的创始人 Alfred Aho.Peter Weinbe ...
- C++11 学习笔记 std::function和bind绑定器
C++11 学习笔记 std::function和bind绑定器 一.std::function C++中的可调用对象虽然具有比较统一操作形式(除了类成员指针之外,都是后面加括号进行调用),但定义方法 ...
- Oracle中RAISE异常
转: Oracle中RAISE异常 由三种方式抛出异常 1. 通过PL/SQL运行时引擎 2. 使用RAISE语句 3. 调用RAISE_APPLICATION_ERROR存储过程 当数据库或PL/S ...
- Spring cloud微服务安全实战-4-9Zuul网关安全开发(二)
把在微服务里面写的安全的相关逻辑挪到网关里面来.这样把安全逻辑和业务逻辑解耦开.那么这些问题就都解决了. 先来看下之前的安全的代码,首先在之类做了认证,认证服务器去认证,拿这个token去换用户信息. ...
- Qt编写自定义控件60-声音波形图
一.前言 这个控件源自于一个音乐播放器,在写该音乐播放器的时候,需要将音频的数据转换成对应的频谱显示,采用的fmod第三方库来处理(fmod声音系统是为游戏开发者准备的革命性音频引擎,非常强大和牛逼) ...
- html如何修改hr水平直线的粗细
hr是常见的超文本标签,是一条水平直线,要设置该直线变粗一些.可以先把hr本身的border隐藏掉,然后设置border-top-width,也就是只留上边框,如图:hr的默认高度height是0,所 ...