"""
一、赋值在python中就是简单的对象引用
"""
list_a = ["aaa", "bbb"]
list_b = list_a
print(id(list_a), id(list_b)) # 输出 2127728239240 2127728239240 # 通过上面操作可以看出,list_b和list_a指向同一片内存,list_b不过是list_a的别名,是引用,除了list_b这个名字以外,没有其它的内存开销。。
# 修改了list_a,就影响了list_b;同理,修改了list_b就影响了list_a list_b[1] = "ccc"
print(f"list_a = {list_a}") # 输出 list_a = ['aaa', 'ccc']
print(f"list_b = {list_b}") # 输出 list_b = ['aaa', 'ccc'] """
二、浅拷贝
在浅拷贝时,拷贝出来的新对象的地址和原对象是不一样的,但是新对象里面的可变元素(如列表)的地址和原对象里的可变元素的地址是相同的,也就是说浅拷贝它拷贝的是浅层次的数据结构(不可变元素),对象里的可变元素作为深层次的数据结构并没有被拷贝到新地址里面去,
而是和原对象里的可变元素指向同一个地址,所以在新对象或原对象里对这个可变元素做修改时,两个对象是同时改变的。
"""
list_c = ["aaa", "bbb"]
list_d = list_c.copy()
print(id(list_c), id(list_d)) # 输出 1803491942920 1803491976008
# 通过id可以看出,发现它们也不指向同一片内存,也就是说浅拷贝产生的list_d不再是list_c了,使用is也可以发现他们不是同一个对象。 for x in list_c:
print(id(x)) # 输出 "aaa" = 1803491959280 "bbb" = 1803491948000
for y in list_d:
print(id(y)) # 输出 "aaa" = 1803491959280 "bbb" = 1803491948000
# 但是当我们查看元素的id时,可以看到两个对象包含的元素的地址是相同的。 list_d[1] = "ccc"
print(f"list_c = {list_c}") # 输出 list_c = ['aaa', 'bbb']
print(f"list_d = {list_d}") # 输出 list_d = ['aaa', 'ccc']
# 在这种情况下,list_c和list_d是不同的对象,修改list_d理论上不会影响list_c。 list_e = [["aaa", "bbb"], "ccc"]
list_f = list_e.copy()
list_f[0][1] = "ccc"
print(f"list_e = {list_e}") # 输出 list_e = [['aaa', 'ccc'], 'ccc']
print(f"list_f = {list_f}") # 输出 list_e = [['aaa', 'ccc'], 'ccc']
# 但是要注意,浅拷贝之所以称为浅拷贝,是它仅仅只拷贝了一层,在list_a中有一个嵌套的list,如果我们修改了它,情况就不一样了。
# 通过list_f[0][1] = "ccc" 修改list_f后,将发现list_e也发生了变化。这是因为,修改嵌套的元素的内存地址是一样,与前面赋值原理一样。

"""
三、深拷贝 深拷贝只有一种形式,copy模块中的deepcopy函数。
和浅拷贝对应,深拷贝拷贝了对象的所有元素,包括多层嵌套的元素。因而,它的时间和空间开销要高。
同样对list_a,若使用list_b = copy.deepcopy(list_a),再修改list_b将不会影响到list_a了。即使嵌套的列表具有更深的层次,也不会产生任何影响。
因为深拷贝出来的对象根本就是一个全新的对象,不再与原来的对象有任何关联。
"""
"""
### 关于拷贝的警告  
1、对于非容器类型,如数字,字符,以及其它“原子”类型,没有拷贝一说。产生的都是原对象的引用。
2、如果元组变量值包含原子类型对象,即使采用了深拷贝,也只能得到浅拷贝。
"""

四、浅拷贝与深拷贝中的注意点

  • 列表的切片以及字典的copy方法都是浅拷贝

  

  

  • 如果对一个全部都是不可变类型的数据用copy.copy()或copy.deepcopy()进行拷贝,他们结果相同,都是引用指向

   

  • 如果拷贝的是一个拥有可变类型的数据,即使元组是最顶层,那么deepcopy依然是 深拷贝,而copy.copy()拷贝的还是指向

  

python的赋值、浅拷贝和深拷贝的更多相关文章

  1. 对Python中列表和数组的赋值,浅拷贝和深拷贝的实例讲解

    引用:https://www.jb51.net/article/142775.htm 列表赋值: 1 2 3 4 5 6 7 >>> a = [1, 2, 3] >>&g ...

  2. 16.python中的浅拷贝和深拷贝

    在讲什么是深浅拷贝之前,我们先来看这样一个现象: a = ['scolia', 123, [], ] b = a[:] b[2].append(666) print a print b

  3. Python 引用、浅拷贝、深拷贝解析

    引用 Python是动态数据类型的语言,故在对变量进行赋值时是不用制定变量类型的. 或者说,你可以把变量赋值的过程,当作是贴一个标签,去引用该数据. 看下面的例子: In [54]: a=4 In [ ...

  4. python 基础之浅拷贝与深拷贝

    浅拷贝与深拷贝 1.普通赋值 例1: a = 15b = aa = 16print(b)#例2:lst1 = [1,2,3]lst2 = lst1lst1.append(4)print(lst2)#这 ...

  5. python 进阶篇 浅拷贝与深拷贝

    阐述引用.浅拷贝和深拷贝前,首先需要要了解 Python 的世界里,一切皆对象,每个对象各包含一个 idendity.type 和 value. 引用(Reference) >>> ...

  6. python中的浅拷贝,深拷贝

    直接引用,间接引用 # 1.列表存储的是索引对应值的内存地址,值会单独的开辟一个内存空间 list = ["a","b"] 内存里面存储的就是list[0],l ...

  7. python中赋值-浅拷贝-深拷贝之间的关系

    赋值: 变量的引用,没有拷贝空间 对象之间赋值本质上 是对象之间的引用传递而已.也就是多个对象指向同一个数据空间. 拷贝的对象分两种类型: . 拷贝可变类型 浅拷贝: 只拷贝第一层数据,不关心里面的第 ...

  8. python中的浅拷贝和深拷贝

    1.赋值语句 a = 'abc' b = a print id(a) print id(b) # id(a):29283464 # id(b):29283464 通过简单的复制,我们可以看到,a b其 ...

  9. Python核心编程--浅拷贝与深拷贝

    一.问题引出浅拷贝 首先看下面代码的执行情况: a = [1, 2, 3] print('a = %s' % a) # a = [1, 2, 3] b = a print('b = %s' % b) ...

  10. Python中的浅拷贝与深拷贝

    编者注:本文主要参考了<Python核心编程(第二版)> 以下都是参考资料后,我自己的理解,如有错误希望大家不吝赐教. 大家有没有遇到这样一种情况,对象赋值后,对其中一个变量进行修改,另外 ...

随机推荐

  1. 【Linux】【Shell】【text】awk

    基本用法:gawk [options] 'program' FILE ...             program: PATTERN{ACTION STATEMENTS}               ...

  2. 【Java 多线程】Java线程池类ThreadPoolExecutor、ScheduledThreadPoolExecutor及Executors工厂类

    Java中的线程池类有两个,分别是:ThreadPoolExecutor和ScheduledThreadPoolExecutor,这两个类都继承自ExecutorService.利用这两个类,可以创建 ...

  3. Servlet+Jdbc+mysql实现登陆功能

    首先是新建一个servlet,servlet中有dopost和doget方法 一般的表格提交都是用post方法,故在dopost里面写入逻辑代码 下面是其逻辑代码Check.java protecte ...

  4. java多线程4:volatile关键字

    上文说到了 synchronized,那么就不得不说下 volatile关键字了,它们两者经常协同处理多线程的安全问题. volatile保证可见性 那么volatile的作用是什么呢? 在jvm运行 ...

  5. Nginx支持php

    目录 一.简介 二.配置 三.测试 四.参数 一.简介 Nginx本身只能解析html文件,但有些网页是php写的,就需要Nginx连接php,将网页解析成html再发给客户端. 配置中将.php 结 ...

  6. 内存泄漏避雷!你真的了解重写equals()和hashcode()方法的原因吗?

    基本概念 要比较两个对象是否相等时需要调用对象的equals() 方法: 判断对象引用所指向的对象地址是否相等 对象地址相等时, 那么对象相关的数据也相等,包括: 对象句柄 对象头 对象实例数据 对象 ...

  7. Python旋转魔方阵

    [问题描述]输入一个自然数N(2≤N≤9),要求输出如下的魔方阵,即边长为N*N,元素取值为1至N*N,1在左上角,呈顺时针方向依次放置各元素. N=3时:    1    2    3    8   ...

  8. CF734B Anton and Digits 题解

    Content 有 \(k_2\) 个 \(2\).\(k_3\) 个 \(3\).\(k_5\) 个 \(5\) 和 \(k_6\) 个 \(6\),你可以用这里面的数字来组成 \(256,32\) ...

  9. CF17A Noldbach problem 题解

    Content 若一个素数可以用比它小的相邻的两个素数的和加 \(1\) 表示,那么称这个素数为"好素数". 给定两个正整数 \(n,k\),问从 \(2\) 到 \(n\) 的好 ...

  10. LuoguP7071 [CSP-J2020] 优秀的拆分 题解

    Content 给定一个数 \(n\),求是否能够拆分成 \(2\) 的正整数次幂的和的形式,并给出具体方案. 数据范围:\(1\leqslant n\leqslant 10^7\). Solutio ...