底层剖析Python深浅拷贝
底层剖析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.保持原本体中的值,不发生变化
其他图示



底层剖析Python深浅拷贝的更多相关文章
- Python开发【第二章】:Python深浅拷贝剖析
Python深浅拷贝剖析 Python中,对象的赋值,拷贝(深/浅拷贝)之间是有差异的,如果使用的时候不注意,就可能产生意外的结果. 下面本文就通过简单的例子介绍一下这些概念之间的差别. 一.对象赋值 ...
- 小学生都能学会的python(深浅拷贝)
小学生都能学会的python(深浅拷贝) join() 把列表中的每一项用字符串拼接起来 # lst = ["汪峰", "吴君如", "李嘉欣&quo ...
- 【0806 | Day 9】三张图带你了解数据类型分类和Python深浅拷贝
一.数据类型分类 二.Python深浅拷贝
- 关于python深浅拷贝的个人浅见
起初,关于python的深浅拷贝,总是习惯去用传值传址的方式去考虑,发现总是get不到规律,容易记混. python有着高度自治的内存管理,而不可变对象的内存分配,则是能省则省,就是说,无论用什么拷贝 ...
- python深浅拷贝与赋值
初学编程的小伙伴都会对于深浅拷贝的用法有些疑问,今天我们就结合python变量存储的特性从内存的角度来谈一谈赋值和深浅拷贝~~~ 预备知识一——python的变量及其存储 在详细的了解python中赋 ...
- python 深浅拷贝 进阶
主要理解新与旧究竟在哪里 这样也就理解了 深浅拷贝 先说说赋值,事实上python中的赋值事实上是赋值了一个引用.比如: foo1=1.0 foo2=foo1 用操作符is推断时.你能够发现结果是tr ...
- python深浅拷贝&垃圾回收&上下文管理(with语句)
深浅拷贝 在Python中使用copy模块用于对象的拷贝操作. 该模块提供了两个主要的方法:浅拷贝 copy.copy() 深拷贝 copy.deepcopy() 1.浅拷贝(copy) 浅拷贝: 不 ...
- 关于Python深浅拷贝
拷贝: 说明:原则上就是把数据分离出来,复制其数据,并以后修改互不影响. 何来深浅拷贝的说法? 深浅拷贝的“深”和“浅”可以理解为从变量到硬盘上的物理存储介质之间的层次的多少. 下面用一个示例来解释浅 ...
- python 深浅拷贝 for循环删除
###########################总结########################### 1. 基础数据类型补充 大多数的基本数据类型的知识.已经学完了 a='aaaa' ls ...
随机推荐
- Rocket - tilelink - TLBusWrapper.to
https://mp.weixin.qq.com/s/jSnhBzU5_ayQCg5fWAcx-g 简单介绍TLBusWrapper.to()的实现.主要介绍确定this{...}对应代码的过程. 1 ...
- Rocket - diplomacy - AddressAdjuster
https://mp.weixin.qq.com/s/X0s5CWN84GEiwpNR7tiRgA 基于AddressAdjuster介绍LazyModule的实现. 参考链接:https://g ...
- TI CC1310 sub1G的SDK开发之入门
CC1310是TI新出的一款sub1G射频模块,具体参数见数据手册吧,这款芯片的SDK跑的是rtos系统,是基于free-rtos定制的ti-rtos,多任务运行.芯片集成了两个核,一个M3做控制MU ...
- Java实现 LeetCode 427 建立四叉树
427. 建立四叉树 我们想要使用一棵四叉树来储存一个 N x N 的布尔值网络.网络中每一格的值只会是真或假.树的根结点代表整个网络.对于每个结点, 它将被分等成四个孩子结点直到这个区域内的值都是相 ...
- Java实现 蓝桥杯VIP 算法提高 计算器
算法提高 计算器 时间限制:1.0s 内存限制:256.0MB [问题描述] 王小二的计算器上面的LED显示屏坏掉了,于是他找到了在计算器维修与应用系学习的你来为他修计算器. 屏幕上可以显示0~9的数 ...
- 第三届蓝桥杯JavaC组国(决)赛真题
解题代码部分来自网友,如果有不对的地方,欢迎各位大佬评论 题目1. 填算式 [结果填空] (满分11分) 看这个算式: ☆☆☆ + ☆☆☆ = ☆☆☆ 如果每个五角星代表 1 ~ 9 的不同的数字. ...
- 11.经典O(n²)比较型排序算法
关注公号「码哥字节」修炼技术内功心法,完整代码可跳转 GitHub:https://github.com/UniqueDong/algorithms.git 摘要:排序算法提多了,很多甚至连名字你都没 ...
- [bx] and loop
1.[bx] 表示一个内存单元,它的偏移地址在bx中 mov al,[bx] 2.描述符号() 来表示一个寄存器或一个内存单元中的内容. 约定符号idata表示常量. 3.loop 标号 CPU在执行 ...
- .NET Core 工作单元unitofwork 实现,基于NPOCO
现有项目中的orm 并非efcore,而是非主流的npoco,本身没有自带工作单元所以需要自己手撸一个,现记录一下,基于其他orm的工作单元照例实现应该没有什么问题 该实现基于NPOCO,针对其他的O ...
- 原生js实现图片瀑布流布局,注释超详细
完整代码: <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF ...