底层剖析Python深浅拷贝

拷贝的用途

  拷贝就是copy,目的在于复制出一份一模一样的数据。使用相同的算法对于产生的数据有多种截然不同的用途时就可以使用copy技术,将copy出的各种副本去做各种不同的操作。

  值得一提的是绝大部分编程语言中对于copy都有深浅拷贝的概念,所以充分的理解本章节的知识也是在为今后学习其他编程语言少走弯路。

  在Python中,除开手动deepcopy(),其他的任何操作都是浅拷贝。

Python = 赋值示例

  好了,废话不多说。直接进入主题,上代码:

>>> # ==== = 赋值示例 ====
>>>
>>> li1 = ["a","b",[1,2]] # 注意存储的数据类型。第一层存储2个不可变类型,1个可变类型小容器(list),第二层存储2个不可变类型
>>> id(li1) # 第一层,查看变量名所指向的列表(第一层容器)的内存地址id号
2324901133120
>>> id(li1[0]) # 第一层,查看第一层容器中的具体元素内存地址id号
2324900663472
>>> id(li1[-1]) # 第二层(小容器id号),查看第一层容器中的小容器list的内存地址id号
2324900812864
>>> li1
['a', 'b', [1, 2]]
>>> # =============手动分割线=============
>>> li2 = li1
>>> id(li2) # 第一层,查看变量名所指向的列表(第一层容器)的内存地址id号
2324901133120
>>> id(li2[0]) # 第一层,查看第一层容器中的具体元素内存地址id号
2324900663472
>>> id(li2[-1]) # 第二层(小容器id号),查看第一层容器中的小容器list的内存地址id号
2324900812864
>>> li2 # 可以看到 li2 与li1 不管是第一层。还是第二层的内存地址id号都全部一样
['a', 'b', [1, 2]]
>>>

  尝试修改li1,查看li2的变化:

>>> li1[0] = "x"
>>> li1[-1][-1] = 20
>>> li1
['x', 'b', [1, 20]]
>>> li2 #可以看到li2随着li1而变化,不管是第一层还是第二层都跟着变化,因为内存引用都全部一样。
['x', 'b', [1, 20]]
>>>

赋值结论:

  本体            

    1.修改不可变类型数据(如:第一层的str

    2.修改可变数据类型中的数据(如:第二层小容器list中的数据)

对应关系

  拷贝体

    1.不保持原本体中的值,跟随本体变化

    2.不保持原本体中的值,跟随本体变化

底层原理


Python 浅拷贝示例

  浅拷贝,用到list数据类型自带的方法,copy()。我们来看一看会怎么样:

>>> # ==== 浅拷贝示例 ====
>>>
>>> li1 = ["a","b",[1,2]] # 注意存储的数据类型。第一层存储2个不可变类型,1个可变类型小容器(list),第二层存储2个不可变类型
>>> id(li1) # 第一层,查看变量名所指向的列表(第一层容器)的内存地址id号
3120558308288
>>> id(li1[0]) # 第一层,查看第一层容器中的具体元素内存地址id号
3120557838512
>>> id(li1[-1]) # 第二层(小容器id号),查看第一层容器中的小容器list的内存地址id号
3120557987904
>>> li1
['a', 'b', [1, 2]]
>>> #=============手动分割线=============
>>> li2 = li1.copy()
>>> id(li2) # 第一层,查看变量名所指向的列表(第一层容器)内存地址id号
3120558308352
>>> id(li2[0]) # 第一层,查看第一层容器中的具体元素内存地址id号
3120557838512
>>> id(li2[-1]) # 第二层(小容器id号),查看第一层容器中的小容器list的内存地址id号
3120557987904
>>> li2 # 可以看到 li2 与li1 第一层的内存地址已经发生了变化。只有第二层的内存引用地址一样
['a', 'b', [1, 2]]
>>>

  尝试修改li1,查看li2的变化:

>>> li1[0] = "x"
>>> li1[-1][-1] = 20
>>> li1
['x', 'b', [1, 20]]
>>> li2 # li2 仅仅只有第二层小容器list中的值发生了变化。而第一层中的str不可变类型并没有发生变化
['a', 'b', [1, 20]]
>>>

浅拷贝结论:

  本体            

    1.修改不可变类型数据(如:第一层的str

    2.修改可变数据类型中的数据(如:第二层小容器list中的数据)

对应关系

  拷贝体

    1.保持原本体中的值,不发生变化

    2.不保持原本体中的值,跟随本体变化

底层原理


Python 深拷贝示例

  使用深拷贝需要导入Python的内置库,copy,具体使用方式还是看代码:

>>> # ==== 深拷贝示例 ====
>>>
>>> from copy import deepcopy # deep深度的意思,copy就拷贝。
>>> li1 = ["a","b",[1,2]] # 注意存储的数据类型。第一层存储2个不可变类型,1个可变类型小容器(list),第二层存储2个不可变 类型
>>> id(li1) # 第一层,查看变量名所指向的列表(第一层容器)的内存地址id号
3120558351168
>>> id(li1[0]) # 第一层,查看第一层容器中的具体元素内存地址id号
3120557838512
>>> id(li1[-1]) # 第二层(小容器id号),查看第一层容器中的小容器list的内存地址id号
3120558353280
>>> li1
['a', 'b', [1, 2]]
>>> #=============手动分割线=============
>>> li2 = deepcopy(li1)
>>> id(li2) # 第一层,查看变量名所指向的列表(第一层容器)内存地址id号
3120558308288
>>> id(li2[0]) # 第一层,查看第一层容器中的具体元素内存地址id号
3120557838512
>>> id(li2[-1]) # 第二层(小容器id号),查看第一层容器中的小容器list的内存地址id号
3120558904448
>>> li2 # 可以看到 li2 与li1 第一层的内存地址已经发生了变化。只有第二层的内存引用地址一样
['a', 'b', [1, 2]]
>>>

  尝试修改li1,查看li2的变化:

>>> li1[0] = "x"
>>> li1[-1][-1] = 20
>>> li1
['x', 'b', [1, 20]]
>>> li2 # li2 由于小容器也新生成了一个。所以即使li1小容器中的值发生改变,li2小容器中的值依然是原本的值
['a', 'b', [1, 2]]
>>>

深拷贝结论:

  本体            

    1.修改不可变类型数据(如:第一层的str

    2.修改可变数据类型中的数据(如:第二层小容器list中的数据)

对应关系

  拷贝体

    1.保持原本体中的值,不发生变化

    2.保持原本体中的值,不发生变化

其他图示

b = a::赋值引用,a 和 b 都指向同一个对象。

b = a.copy():浅拷贝, a 和 b 是一个独立的对象,但他们的子对象还是指向统一对象(是引用)。

b = copy.deepcopy(a): 深度拷贝, a 和 b 完全拷贝了父对象及其子对象,两者是完全独立的。

底层剖析Python深浅拷贝的更多相关文章

  1. Python开发【第二章】:Python深浅拷贝剖析

    Python深浅拷贝剖析 Python中,对象的赋值,拷贝(深/浅拷贝)之间是有差异的,如果使用的时候不注意,就可能产生意外的结果. 下面本文就通过简单的例子介绍一下这些概念之间的差别. 一.对象赋值 ...

  2. 小学生都能学会的python(深浅拷贝)

    小学生都能学会的python(深浅拷贝) join() 把列表中的每一项用字符串拼接起来 # lst = ["汪峰", "吴君如", "李嘉欣&quo ...

  3. 【0806 | Day 9】三张图带你了解数据类型分类和Python深浅拷贝

    一.数据类型分类 二.Python深浅拷贝

  4. 关于python深浅拷贝的个人浅见

    起初,关于python的深浅拷贝,总是习惯去用传值传址的方式去考虑,发现总是get不到规律,容易记混. python有着高度自治的内存管理,而不可变对象的内存分配,则是能省则省,就是说,无论用什么拷贝 ...

  5. python深浅拷贝与赋值

    初学编程的小伙伴都会对于深浅拷贝的用法有些疑问,今天我们就结合python变量存储的特性从内存的角度来谈一谈赋值和深浅拷贝~~~ 预备知识一——python的变量及其存储 在详细的了解python中赋 ...

  6. python 深浅拷贝 进阶

    主要理解新与旧究竟在哪里 这样也就理解了 深浅拷贝 先说说赋值,事实上python中的赋值事实上是赋值了一个引用.比如: foo1=1.0 foo2=foo1 用操作符is推断时.你能够发现结果是tr ...

  7. python深浅拷贝&垃圾回收&上下文管理(with语句)

    深浅拷贝 在Python中使用copy模块用于对象的拷贝操作. 该模块提供了两个主要的方法:浅拷贝 copy.copy() 深拷贝 copy.deepcopy() 1.浅拷贝(copy) 浅拷贝: 不 ...

  8. 关于Python深浅拷贝

    拷贝: 说明:原则上就是把数据分离出来,复制其数据,并以后修改互不影响. 何来深浅拷贝的说法? 深浅拷贝的“深”和“浅”可以理解为从变量到硬盘上的物理存储介质之间的层次的多少. 下面用一个示例来解释浅 ...

  9. python 深浅拷贝 for循环删除

    ###########################总结########################### 1. 基础数据类型补充 大多数的基本数据类型的知识.已经学完了 a='aaaa' ls ...

随机推荐

  1. Parrot os笔记本推荐

    parrot os基于debian开发的,因此同样适用于其他linux:笔记本集显最好,linux直接适用于intel,不用手动切换显卡,大多数linux玩家及pentester不需要高性能显卡,当然 ...

  2. 说了这么多次 I/O,但你知道它的原理么

    O 软件目标 设备独立性 现在让我们转向对 I/O 软件的研究,I/O 软件设计一个很重要的目标就是设备独立性(device independence).啥意思呢?这意味着我们能够编写访问任何设备的应 ...

  3. 【Hadoop】配置全分布式模式

    分布式原理 配置 详细过程 假设有三台虚拟机,1台master主机namenode,2台slave奴隶机datanode 所有机器都要配好jdk.Java环境变量.hadoop_env.sh里java ...

  4. Attribute (XXX) is obsolete. Its use is discouraged in HTML5 documents.

    这种警告主要是因为这些属性在HTML5中过时了,并不影响代码运行,但是一些强迫症就会非常难受. 解决办法: 将程序的顶部的这句: !DOCTYPE 修改为: !DOCTYPE html PUBLIC ...

  5. JVM虚拟机 与 GC 垃圾回收

    个人博客网:https://wushaopei.github.io/    (你想要这里多有) 一.JVM体系结构概述 1.JVM 与系统.硬件 ​ JVM是运行在操作系统之上的,它与硬件没有直接的交 ...

  6. 高性能可扩展mysql 笔记(六) SQL执行计划及分页查询优化、分区键统计

    个人博客网:https://wushaopei.github.io/    (你想要这里多有) 常见业务处理 一.使用数据库处理常见业务: 案例: 如何对评论进行分页展示 使用 EXPLAIN 获得s ...

  7. Java实现 LeetCode 775 全局倒置与局部倒置(分析题)

    775. 全局倒置与局部倒置 数组 A 是 [0, 1, -, N - 1] 的一种排列,N 是数组 A 的长度.全局倒置指的是 i,j 满足 0 <= i < j < N 并且 A ...

  8. Java实现 LeetCode 695 岛屿的最大面积(DFS)

    695. 岛屿的最大面积 给定一个包含了一些 0 和 1 的非空二维数组 grid . 一个 岛屿 是由一些相邻的 1 (代表土地) 构成的组合,这里的「相邻」要求两个 1 必须在水平或者竖直方向上相 ...

  9. Java实现 LeetCode 687 最长同值路径(递归)

    687. 最长同值路径 给定一个二叉树,找到最长的路径,这个路径中的每个节点具有相同值. 这条路径可以经过也可以不经过根节点. 注意:两个节点之间的路径长度由它们之间的边数表示. 示例 1: 输入: ...

  10. Java实现蓝桥杯墓地雕塑

    墓地雕塑 问题描述 在一个周长为10000的圆上等距分布着n个雕塑.现在又有m个新雕塑加入(位置可以随意放), 希望所有n+m个雕塑在圆周上均匀分布.这就需要移动其中一些原有的雕塑.要求n个雕塑移动的 ...