底层剖析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 ...
随机推荐
- DDD之2领域概念
图中是暗黑领域,非常牛逼的技能. 背景 DDD中出现的名词: 领域,子领域,核心域,通用域,支撑域,限界上下文,聚合,聚合根,实体,值对象 都是关键概念,但是又比较晦涩,在开始DDD之前,搞清楚这些关 ...
- burpsuite 关于部分https抓包失败原因
没导入证书 burpsuite生成证书 der格式,名字随便取,一路next firefox浏览器导入 导入,勾选信任证书ok,重启浏览器 还有你要勾选这里,确保所有流量都走你的代理 ps:遇到浏 ...
- 【Hadoop】配置全分布式模式
分布式原理 配置 详细过程 假设有三台虚拟机,1台master主机namenode,2台slave奴隶机datanode 所有机器都要配好jdk.Java环境变量.hadoop_env.sh里java ...
- 基础数论——EXGCD
1.前言 \(皆さん.こんにちは.\)今天我们来讲 \(EXGCD\) .(扩展欧几里得) 既然是扩展嘛,那肯定有不扩展的,也就是 \(GCD\) . 我们都知道 \(GCD\) 怎么写: ll GC ...
- @Transactional 注解失效场景
@Transactional可以用在接口.类.类方法上. 作用于类:当把@Transactional注解放在类上时,表示该类的所有public方法都配置了该事物注解. 作用于方法:表示该方法配置了事物 ...
- Java实现 LeetCode 461 汉明距离
461. 汉明距离 两个整数之间的汉明距离指的是这两个数字对应二进制位不同的位置的数目. 给出两个整数 x 和 y,计算它们之间的汉明距离. 注意: 0 ≤ x, y < 231. 示例: 输入 ...
- Java实现 LeetCode 394 字符串解码
394. 字符串解码 给定一个经过编码的字符串,返回它解码后的字符串. 编码规则为: k[encoded_string],表示其中方括号内部的 encoded_string 正好重复 k 次.注意 k ...
- Java实现蓝桥杯3n+1问题
3n+1 [问题描述] 考虑如下的序列生成算法:从整数 n 开始,如果 n 是偶数,把它除以 2:如果 n 是奇数,把它乘 3 加1.用新得到的值重复上述步骤,直到 n = 1 时停止.例如,n = ...
- java实现取球博弈
今盒子里有n个小球,A.B两人轮流从盒中取球,每个人都可以看到另一个人取了多少个,也可以看到盒中还剩下多少个,并且两人都很聪明,不会做出错误的判断. 我们约定: 每个人从盒子中取出的球的数目必须是:1 ...
- Java实现第八届蓝桥杯字母组串
字母组串 由 A,B,C 这3个字母就可以组成许多串. 比如:"A","AB","ABC","ABA","AA ...