深浅复制详解

2024-05-02 16:38
文章标签 详解 复制 深浅

本文主要是介绍深浅复制详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

浅复制:

复制列表,默认做的就是浅复制。如果所有元素都是不可变的,那么使用浅复制没有问题,还能节省内存。

问题如下:列表元素是可变的,复制时,默认做浅复制。list2中的列表和list1中的列表指向同一个列表,元组也是。当对list1或2中的列表进行删除或追加的时候,操作的是同一个引用对象。元组+=操作则会创建一个新元组。
list1 = [3, [66,44, 55], (7, 8, 9)]
list2 = list(list1)
list1.append(100)   #对list2没有影响
print('1:', list1)
print('2:', list2)
list1[1].remove(55)   #对list2有影响,因为list2[1]与list1[1]绑定的是同一个列表。
print('1:', list1)
print('2:', list2)
list2[1] += [33, 22]  # 就地操作
list2[2] += (10, 11)  # 元组 += 运算符会创建一个新元组
print('1:', list1)
print('2:', list2)
结果:
1: [3, [66, 44, 55], (7, 8, 9), 100]
2: [3, [66, 44, 55], (7, 8, 9)]
1: [3, [66, 44], (7, 8, 9), 100]
2: [3, [66, 44], (7, 8, 9)]
1: [3, [66, 44, 33, 22], (7, 8, 9), 100]
2: [3, [66, 44, 33, 22], (7, 8, 9, 10, 11)]

list2复制list1后状态如下:

上图是执行 list2 = list(l1) 赋值后的程序状态。list1 和 list2指代不同的列表,但是二者引用同一个列表 [66, 55, 44] 和元组(7, 8, 9)
 

上图是 list1 和 list2 的最终状态: 二者依然引用同一个列表对象, 现在列表的值是 [66, 44, 33, 22], 不过 list2[2] += (10, 11) 创建一个新元组, 内容是 (7, 8, 9, 10, 11), 它与 list1[2] 引用的元组(7, 8, 9) 无关。

深复制

浅复制没什么问题,但有时我们需要的是深复制(即副本不共享内部对象的引用)。copy模块提供的deepcopycopy函数能为任何对象做深复制和浅复制。

数据完全不共享(复制其数据完完全全放独立的一个内存,完全拷贝,数据不共享)。

 深拷贝就是完完全全复制了一份,且数据不会互相影响,因为内存不共享。

class Bus:def __init__(self, passengers=None):if passengers is None:self.passengers = []else:self.passengers = list(passengers)def pick(self, name):self.passengers.append(name)def drop(self, name):self.passengers.remove(name)import copybus1 = Bus(['alice', 'alex', 'bill', 'david'])
bus2 = copy.copy(bus1)  # 浅复制
bus3 = copy.deepcopy(bus1)  # 深复制
print(id(bus1), id(bus2), id(bus3))  # 685543908520 685543910032 685543910704
bus1.drop('bill')
print(bus1.passengers)  # ['alice', 'alex', 'david']
print(bus2.passengers)  # ['alice', 'alex', 'david']
print(id(bus1.passengers), id(bus2.passengers),id(bus3.passengers))  # 685543921608 685543921608 685543842632,bus1和bus2共享同一个列表对象
print(bus3.passengers)  # ['alice', 'alex', 'bill', 'david']

 

这篇关于深浅复制详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/954697

相关文章

Python中局部变量和全局变量举例详解

《Python中局部变量和全局变量举例详解》:本文主要介绍如何通过一个简单的Python代码示例来解释命名空间和作用域的概念,它详细说明了内置名称、全局名称、局部名称以及它们之间的查找顺序,文中通... 目录引入例子拆解源码运行结果如下图代码解析 python3命名空间和作用域命名空间命名空间查找顺序命名空

SpringRetry重试机制之@Retryable注解与重试策略详解

《SpringRetry重试机制之@Retryable注解与重试策略详解》本文将详细介绍SpringRetry的重试机制,特别是@Retryable注解的使用及各种重试策略的配置,帮助开发者构建更加健... 目录引言一、SpringRetry基础知识二、启用SpringRetry三、@Retryable注解

springboot项目中常用的工具类和api详解

《springboot项目中常用的工具类和api详解》在SpringBoot项目中,开发者通常会依赖一些工具类和API来简化开发、提高效率,以下是一些常用的工具类及其典型应用场景,涵盖Spring原生... 目录1. Spring Framework 自带工具类(1) StringUtils(2) Coll

Python中的魔术方法__new__详解

《Python中的魔术方法__new__详解》:本文主要介绍Python中的魔术方法__new__的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、核心意义与机制1.1 构造过程原理1.2 与 __init__ 对比二、核心功能解析2.1 核心能力2.2

在PyCharm中安装PyTorch、torchvision和OpenCV详解

《在PyCharm中安装PyTorch、torchvision和OpenCV详解》:本文主要介绍在PyCharm中安装PyTorch、torchvision和OpenCV方式,具有很好的参考价值,... 目录PyCharm安装PyTorch、torchvision和OpenCV安装python安装PyTor

SpringBoot条件注解核心作用与使用场景详解

《SpringBoot条件注解核心作用与使用场景详解》SpringBoot的条件注解为开发者提供了强大的动态配置能力,理解其原理和适用场景是构建灵活、可扩展应用的关键,本文将系统梳理所有常用的条件注... 目录引言一、条件注解的核心机制二、SpringBoot内置条件注解详解1、@ConditionalOn

Qt spdlog日志模块的使用详解

《Qtspdlog日志模块的使用详解》在Qt应用程序开发中,良好的日志系统至关重要,本文将介绍如何使用spdlog1.5.0创建满足以下要求的日志系统,感兴趣的朋友一起看看吧... 目录版本摘要例子logmanager.cpp文件main.cpp文件版本spdlog版本:1.5.0采用1.5.0版本主要

Linux ls命令操作详解

《Linuxls命令操作详解》通过ls命令,我们可以查看指定目录下的文件和子目录,并结合不同的选项获取详细的文件信息,如权限、大小、修改时间等,:本文主要介绍Linuxls命令详解,需要的朋友可... 目录1. 命令简介2. 命令的基本语法和用法2.1 语法格式2.2 使用示例2.2.1 列出当前目录下的文

MySQL中的交叉连接、自然连接和内连接查询详解

《MySQL中的交叉连接、自然连接和内连接查询详解》:本文主要介绍MySQL中的交叉连接、自然连接和内连接查询,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、引入二、交php叉连接(cross join)三、自然连接(naturalandroid join)四

Go 语言中的select语句详解及工作原理

《Go语言中的select语句详解及工作原理》在Go语言中,select语句是用于处理多个通道(channel)操作的一种控制结构,它类似于switch语句,本文给大家介绍Go语言中的select语... 目录Go 语言中的 select 是做什么的基本功能语法工作原理示例示例 1:监听多个通道示例 2:带