深浅复制详解

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

相关文章

C++使用栈实现括号匹配的代码详解

《C++使用栈实现括号匹配的代码详解》在编程中,括号匹配是一个常见问题,尤其是在处理数学表达式、编译器解析等任务时,栈是一种非常适合处理此类问题的数据结构,能够精确地管理括号的匹配问题,本文将通过C+... 目录引言问题描述代码讲解代码解析栈的状态表示测试总结引言在编程中,括号匹配是一个常见问题,尤其是在

Debezium 与 Apache Kafka 的集成方式步骤详解

《Debezium与ApacheKafka的集成方式步骤详解》本文详细介绍了如何将Debezium与ApacheKafka集成,包括集成概述、步骤、注意事项等,通过KafkaConnect,D... 目录一、集成概述二、集成步骤1. 准备 Kafka 环境2. 配置 Kafka Connect3. 安装 D

Java中ArrayList和LinkedList有什么区别举例详解

《Java中ArrayList和LinkedList有什么区别举例详解》:本文主要介绍Java中ArrayList和LinkedList区别的相关资料,包括数据结构特性、核心操作性能、内存与GC影... 目录一、底层数据结构二、核心操作性能对比三、内存与 GC 影响四、扩容机制五、线程安全与并发方案六、工程

Spring Cloud LoadBalancer 负载均衡详解

《SpringCloudLoadBalancer负载均衡详解》本文介绍了如何在SpringCloud中使用SpringCloudLoadBalancer实现客户端负载均衡,并详细讲解了轮询策略和... 目录1. 在 idea 上运行多个服务2. 问题引入3. 负载均衡4. Spring Cloud Load

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

在 Spring Boot 中使用 @Autowired和 @Bean注解的示例详解

《在SpringBoot中使用@Autowired和@Bean注解的示例详解》本文通过一个示例演示了如何在SpringBoot中使用@Autowired和@Bean注解进行依赖注入和Bean... 目录在 Spring Boot 中使用 @Autowired 和 @Bean 注解示例背景1. 定义 Stud

如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别详解

《如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别详解》:本文主要介绍如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别的相关资料,描述了如何使用海康威视设备网络SD... 目录前言开发流程问题和解决方案dll库加载不到的问题老旧版本sdk不兼容的问题关键实现流程总结前言作为

SQL 中多表查询的常见连接方式详解

《SQL中多表查询的常见连接方式详解》本文介绍SQL中多表查询的常见连接方式,包括内连接(INNERJOIN)、左连接(LEFTJOIN)、右连接(RIGHTJOIN)、全外连接(FULLOUTER... 目录一、连接类型图表(ASCII 形式)二、前置代码(创建示例表)三、连接方式代码示例1. 内连接(I

Go路由注册方法详解

《Go路由注册方法详解》Go语言中,http.NewServeMux()和http.HandleFunc()是两种不同的路由注册方式,前者创建独立的ServeMux实例,适合模块化和分层路由,灵活性高... 目录Go路由注册方法1. 路由注册的方式2. 路由器的独立性3. 灵活性4. 启动服务器的方式5.

Java中八大包装类举例详解(通俗易懂)

《Java中八大包装类举例详解(通俗易懂)》:本文主要介绍Java中的包装类,包括它们的作用、特点、用途以及如何进行装箱和拆箱,包装类还提供了许多实用方法,如转换、获取基本类型值、比较和类型检测,... 目录一、包装类(Wrapper Class)1、简要介绍2、包装类特点3、包装类用途二、装箱和拆箱1、装