一. 赋值运算

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的更多相关文章

  1. 『Python基础』第2节: Python简介及入门

    一. Python介绍 Python是一门高级计算机程序设计语言,1989年,荷兰的Guido von Rossum创造了它.Guido是是一个牛人,1982年,他从阿姆斯特丹大学获得了数学和计算机硕 ...

  2. 『Python基础』第4节:基础数据类型初识

    本节只是对基础数据类型做个简单介绍, 详情会在之后慢慢介绍 什么是数据类型? 我们人类可以分清数字与字符串的区别, 可是计算机不能. 虽然计算机很强大, 但在某种程度上又很傻, 除非你明确告诉它数字与 ...

  3. 『Python基础』第7节:基本运算符

    一. 基本运算符 运算按种类可以分为: 算数运算.比较运算.逻辑运算.赋值运算.成员运算.身份运算.位运算. 今天我们只学习算数运算.比较运算.逻辑运算.赋值运算.成员运算 1.1 算数运算 以下假设 ...

  4. 『Python基础』第5节:条件控制

    if 语句的使用 单分支 if 条件: 满足条件后要执行的代码 例如: if 2 < 3: print(222) print(333) 每个条件后面都要使用冒号 :, 表示接下来是满足条件后要执 ...

  5. 『Python基础』第39节 函数的返回值

    1. 函数的返回值 ​ 一个函数就是封装一个功能, 这个功能一般都会有一个最终结果的. ​ 比如写一个登录的函数, 最终登录是否成功你总得告诉我一声吧? ​ 还有咱们之前也用过 len() 这个函数, ...

  6. 『Python基础』第8节:格式化输出

    现在有一个需求, 询问用户的姓名, 年龄, 工作, 爱好, 然后打印成以下格式 ************ info of Conan ************ name: Conan age: 23 ...

  7. 『Python基础』第6节:流程控制之while循环

    在生活中经常遇到循环的事情, 比如循环列表播放歌曲等. 在Python中, 也有循环, 就是其流程控制语句while. 1. 基本循环 while 条件: 循环体 # 如果条件为真, 那么就执行循环体 ...

  8. 『Python基础』第1节 Windows环境下安装Python3.x

    一. Python安装 1. 下载安装包 https://www.python.org/downloads/release/python-374/ # 3.7安装包 # 如需安装python2.7版本 ...

  9. 『Go基础』第8节 格式化输出

    输出就是将数据信息打印到电脑屏幕上. 本节我们就来学习一下Go语言中的三种输出方式: Print().Println().Printf(). 1.Print() Print()主要的一个特点就是打印数 ...

随机推荐

  1. sass登陆页面实例

    sass登陆页面实例 一.总结 一句话总结: sass使用非常方便:使用就是将sass转化为css引入,并且动态监听让sass转化为css,可以很方便的所见即所得 1.sass安装? npm就可以按照 ...

  2. iview3 版本 升级

    Button 废弃 type ghost,原先的 default 样式有改变. Icon 的图标升级至 ionicons 3.0 图标,图标名称有改变. Breadcrumb 废弃 href 属性. ...

  3. kubernetes 亲和性调度详解

    文章目录 1 概述: 2 场景一:调度到一组具有相同特性的主机上(label+nodeSelector) 3 场景二:部署的应用不想调度到某些节点上(nodeaffinity) 4 场景三:部署的应用 ...

  4. openresty开发系列2--nginx的简单安装,正向、反向代理及常用命令和信号控制介绍

    openresty开发系列2--nginx的简单安装,正向.反向代理及常用命令和信号控制介绍 一.nginx的安装下载及编译安装1.Nginx下载:nginx-1.13.0.tar.gz,下载到:/u ...

  5. shell编程系列14--文本处理三剑客之awk的概述及常用方法总结

    shell编程系列14--文本处理三剑客之awk的概述及常用方法总结 awk是一个文本处理工具,通常用于处理数据并生成结果报告 awk的命名是它的创始人 Alfred Aho.Peter Weinbe ...

  6. C++11 学习笔记 std::function和bind绑定器

    C++11 学习笔记 std::function和bind绑定器 一.std::function C++中的可调用对象虽然具有比较统一操作形式(除了类成员指针之外,都是后面加括号进行调用),但定义方法 ...

  7. Oracle中RAISE异常

    转: Oracle中RAISE异常 由三种方式抛出异常 1. 通过PL/SQL运行时引擎 2. 使用RAISE语句 3. 调用RAISE_APPLICATION_ERROR存储过程 当数据库或PL/S ...

  8. Spring cloud微服务安全实战-4-9Zuul网关安全开发(二)

    把在微服务里面写的安全的相关逻辑挪到网关里面来.这样把安全逻辑和业务逻辑解耦开.那么这些问题就都解决了. 先来看下之前的安全的代码,首先在之类做了认证,认证服务器去认证,拿这个token去换用户信息. ...

  9. Qt编写自定义控件60-声音波形图

    一.前言 这个控件源自于一个音乐播放器,在写该音乐播放器的时候,需要将音频的数据转换成对应的频谱显示,采用的fmod第三方库来处理(fmod声音系统是为游戏开发者准备的革命性音频引擎,非常强大和牛逼) ...

  10. html如何修改hr水平直线的粗细

    hr是常见的超文本标签,是一条水平直线,要设置该直线变粗一些.可以先把hr本身的border隐藏掉,然后设置border-top-width,也就是只留上边框,如图:hr的默认高度height是0,所 ...