Python对象复制竟然有这么多种方式,赶紧学起来!

目录

1、浅拷贝:copy模块的copy()函数 📋

1.1 浅拷贝原理揭秘

1.2 实战演示:列表与字典的浅拷贝

列表浅拷贝示例

字典浅拷贝示例

1.3 注意事项:共享引用与独立对象

2、深拷贝:copy模块的deepcopy()函数 📌

2.1 深拷贝实现机制解析

2.2 深拷贝优势分析

2.3 深度遍历复制实例

列表深度拷贝示例

字典深度拷贝示例

2.4 陷阱规避:彻底分离原对象

3、列表切片法 🍏

3.1 简单快捷的列表复制技巧

示例代码

3.2 切片法与其他方法性能对比

3.3 应用场景与限制条件

应用场景

限制条件

4、对象构造器复刻 🔄

4.1 通过类的构造函数创建新实例

示例代码:基本对象复刻

4.2 应对有自定义__init__方法的对象

示例代码:带克隆方法的复刻

4.3通过类方法克隆对象

实现浅拷贝

实现深拷贝

4.4 自定义复制逻辑

5、序列化与反序列化:pickle模块 🔍

5.1 对象的存储与恢复

序列化(Pickling)

反序列化(Unpickling)

5.2 性能考量与安全提示

性能方面

安全提示

6、不可变类型复制:无需担忧 🙅‍♀️

6.1 字符串、元组的“复制”本质

字符串复制

元组复制

6.2 为何不可变对象“复制”无需担忧

实践意义

6.3 效率与内存占用分析

7、高级话题:numpy数组的复制 📊

7.1 view与copy的区别

View(视图)

Copy(拷贝)

7.2 高性能科学计算中的应用

8、总结与最佳实践 🏆



1、浅拷贝:copy模块的copy()函数 📋

在Python中,对象的复制是一个常见需求,尤其是在处理复杂数据结构时。浅拷贝通过copy模块的copy()函数实现,它创建原始对象的一个新实例,但这个新实例内部的子对象仍然是对原对象子对象的引用。这节将深入探讨浅拷贝的工作原理,并通过实战示例展示其在列表与字典中的应用,同时指出使用过程中需要注意的共享引用问题。

1.1 浅拷贝原理揭秘

浅拷贝的核心在于创建一个新对象,该对象的顶层结构是原始对象的精确副本,但对于嵌套的可变对象(如列表内的列表、字典内的字典等),新旧对象将共享这些嵌套对象的引用。这意味着,修改新对象中的顶层元素不会影响原对象,但直接修改嵌套对象则会影响两个对象。

1.2 实战演示:列表与字典的浅拷贝

列表浅拷贝示例
import copyoriginal_list = [1, 2, [3, 4]]
shallow_copy_list = copy.copy(original_list)print("Original List:", original_list)
print("Shallow Copy List:", shallow_copy_list)original_list[2][0] = 'Modified'
print("\nAfter modifying nested element in original list:")
print("Original List:", original_list)
print("Shallow Copy List:", shallow_copy_list)

输出结果:

Original List: [1, 2, [3, 4]]
Shallow Copy List: [1, 2, [3, 4]]After modifying nested element in original list:
Original List: [1, 2, ['Modified', 4]]
Shallow Copy List: [1, 2, ['Modified', 4]]

从输出可见 ,修改原列表中的嵌套列表元素后,浅拷贝列表中的对应元素也随之改变,体现了浅拷贝的特性。

字典浅拷贝示例
original_dict = {'a': 1, 'b': {'c': 2}}
shallow_copy_dict = copy.copy(original_dict)print("Original Dict:", original_dict)
print("Shallow Copy Dict:", shallow_copy_dict)original_dict['b']['c'] = 'Altered'
print("\nAfter altering nested value in original dict:")
print("Original Dict:", original_dict)
print("Shallow Copy Dict:", shallow_copy_dict)

输出结果:

Original Dict: {'a': 1, 'b': {'c': 2}}
Shallow Copy Dict: {'a': 1, 'b': {'c': 2}}After altering nested value in original dict:
Original Dict: {'a': 1, 'b': {'c': 'Altered'}}
Shallow Copy Dict: {'a': 1, 'b': {'c': 'Altered'}}

同样地 ,字典中嵌套字典的修改也影响到了浅拷贝字典的对应内容 ,再次证明了浅拷贝的浅层次复制特性。

1.3 注意事项:共享引用与独立对象

使用copy.copy()时,重要的是要意识到哪些对象会被共享引用。对于不可变类型(如字符串、数字、元组) ,由于它们不可更改,所以无论深浅拷贝都不会有问题。然而 ,对于可变类型的嵌套结构 ,开发者必须小心 ,因为浅拷贝可能导致意外的数据同步。

为了避免这种情况 ,当需要完全复制一个包含多层嵌套结构的对象时,应考虑使用copy.deepcopy()函数进行深拷贝,确保每个层级的对象都是独立的新实例。

2、深拷贝:copy模块的deepcopy()函数 📌

在Python中处理复杂数据结构时,浅拷贝可能不足以满足完全独立复制对象的需求 ,此时copy模块提供的deepcopy()函数便显得尤为重要。这一章将深入探讨深拷贝的优势、具体实现过程 ,并阐述如何有效避免因对象引用带来的潜在陷阱。

2.1 深拷贝实现机制解析

copy.deepcopy()函数通过递归地遍历对象的所有层次来实现深拷贝。这一过程包括以下几个步骤:

  • • 初始化:检查待拷贝对象的类型,确定拷贝策略。

  • • 递归拷贝:对于简单数据类型 ,直接复制值;对于复杂类型(如列表、字典) ,创建新容器,并递归地对每个元素调用deepcopy()

  • • 处理循环引用:在递归过程中,通过维护一张“备忘录”字典记录已拷贝过的对象 ,遇到相同的引用时直接返回已拷贝的对象,避免无限循环。

2.2 深拷贝优势分析

深拷贝通过递归地复制原对象及其所有子对象,确保新旧对象之间完全独立,即使嵌套结构中的元素也不会共享。这种机制提供了几个关键优势:

  • • 数据隔离:修改深拷贝后的对象不会影响原对象,这对于需要保护原始数据完整性或进行并行处理的场景极为关键。

  • • 一致性控制:在多线程或并发环境下,深拷贝可以减少竞态条件,确保每个线程处理的是数据的一个独立视图。

  • • 模型复用与实验:在构建和测试算法模型时,深拷贝允许用户基于同一初始状态快速创建多个可独立修改的实例。

2.3 深度遍历复制实例

列表深度拷贝示例
import copyoriginal_list = [1, 2, [3, 4]]
deep_copy_list = copy.deepcopy(original_list)# 修改顶层元素
deep_copy_list[0] = 'X'
print("修改后 deep_copy_list:", deep_copy_list)
print("原列表 original_list:", original_list)# 修改嵌套列表
deep_copy_list[2][0] = 'Y'
print("再次修改后 deep_copy_list:", deep_copy_list)
print("原列表 original_list不受影响:", original_list)

通过对比输出,我们可以观察到深拷贝后,无论顶层还是嵌套层次的修改都不会影响原对象。

字典深度拷贝示例
original_dict = {'a': 1, 'b': {'c': 2}}
deep_copy_dict = copy.deepcopy(original_dict)# 修改顶层键值
deep_copy_dict['a'] = 'A'
print("修改后 deep_copy_dict:", deep_copy_dict)
print("原字典 original_dict:", original_dict)# 修改嵌套字典
deep_copy_dict['b']['c'] = 'C'
print("再次修改后 deep_copy_dict:", deep_copy_dict)
print("原字典 original_dict不受影响:", original_dict)

同样,字典的深拷贝保证了即使是嵌套结构中的改变也不会波及原字典。

2.4 陷阱规避:彻底分离原对象

尽管深拷贝提供了强大的独立性,但在处理循环引用或特殊对象(如文件句柄、数据库连接等)时,仍需谨慎:

  • • 循环引用:深拷贝能够处理简单的循环引用,但复杂的循环结构可能导致效率降低或栈溢出。在设计数据结构时,应尽量避免不必要的循环引用。

  • • 非纯数据对象:对于具有副作用或状态的对象(如文件、网络连接、数据库游标等) ,深拷贝可能无法正确复制其状态或行为,这时可能需要自定义复制逻辑。

通过深入理解深拷贝的原理与实践,开发者能够更有效地管理对象生命周期,避免意料之外的数据交互 ,从而提升代码的稳定性和可维护性。

3、列表切片法 🍏

列表切片是Python中一种简洁而高效的方式,用于快速复制列表或其部分元素。这种方法不仅代码简洁,执行速度快 ,而且无需引入额外模块,是处理简单复制任务的理想选择。

3.1 简单快捷的列表复制技巧

切片法是Python中一种快速且简洁的复制列表方式。其基本语法是利用索引切片操作,即使不指定开始和结束索引,也能实现列表的完全复制。具体来说,list[:]即可创建原列表的一个全新副本。这种方法

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/352503.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

(微服务实战)预付卡平台支付交易系统消费业务流程设计

1 交易系统技术架构 预付卡支付交易系统采用Dubbo3作为底层框架,支付交易系统分为账户系统、清结算系统、支付网关、核心支付系统等模块。系统整体采用微服务架构,容器化部署。 2 消费业务流程设计 预付卡系统消费场景分为线上和线下,线…

【安装和引入 PyTorch 包,快来收藏】

在本文介绍 PyTorch 中一些最常用的命令和设置。 一个完成的 PyTorch 工作流程。 安装和引入 PyTorch 包 最好的安装教程就是去官方网站:https://pytorch.org/get-started/locally/ 安装结束之后,直接引入整个 torch 包: import torch或…

【机器学习300问】122、RNN面临哪些问题?

循环神经网络(RNN)主要面临梯度消失和梯度爆炸两个核心问题,这严重影响了其处理长期依赖的能力。此外,还存在一些其他的技术挑战。 一、两个主要问题 (1)梯度消失和梯度爆炸问题 这是RNN中最显著的问题之…

示例:WPF中绑定枚举到ComboBox的方式

一、目的:在开发过程中,经常会需要把枚举绑定到ComboxBox下拉列表中,其实方法有很多,这里面通过MarkupExtension扩展GetEnumSourceExtension去绑定到列表 二、实现 定义GetEnumSourceExtension类 public class GetEnumSourceExte…

Elasticsearch-使用Logstash同步Mysql

1.安装logstash es服务器版本必须和logstash版本一致 7.9.2 在/usr/local/src/下新建logstash文件夹,解压 下载logstash后查看是否安装成功,在logstash的bin目录下输入指令: ./logstash -e input { stdin { } } output { stdout {} }2.my…

【乳业巨擘·数字革命先锋】光明乳业:上市公司科技蜕变,搭贝低代码引领未来新纪元

在这个由科技编织的未来世界里,光明乳业股份有限公司以巨人之姿,傲立于乳业之巅,以其无与伦比的胆识与魄力,引领了一场震撼业界的数字化革命。与低代码领域的创新领袖——搭贝的强强联合,不仅标志着光明乳业在数字化转…

吉林省教育学院学报杂志社吉林省教育学院学报编辑部2024年第5期目录

“研培一体”理论与实践 教师培训管理共同体的职能定位与价值追求 张岩; 1-3 数字化转型背景下教师培训工作的发展路径 李春光; 4-6 挖掘数智潜能,推进教师培训融合创新 鲍赫; 7-9《吉林省教育学院学报》投稿:cn7kantougao163.com 精准培…

AcWing 1273:天才的记忆 ← ST算法求解RMQ问题

【题目来源】https://www.acwing.com/problem/content/1275/【题目描述】 从前有个人名叫 WNB,他有着天才般的记忆力,他珍藏了许多许多的宝藏。 在他离世之后留给后人一个难题(专门考验记忆力的啊!),如果谁…

RockChip Android12 Settings一级菜单

一:概述 在之前的文章中对Android8.1 Settings的流程进行了说明,本章将针对Android12 Settings一级菜单的加载逻辑进行详细说明,Settings版本之间的差异不是很大,有兴趣的同学可自行学习,本文不在做赘述。 Android8.1 Settings说明:RockChip Android8.1 Settings-CSDN博…

创邻科技张晨:期待解锁图技术在供应链中的关联力

近日,创邻科技创始人兼CEO张晨博士受浙江省首席信息官协会邀请,参加数字化转型与企业出海研讨会。 此次研讨会旨在深入探讨数字经济时代下,企业如何有效应对成本提升与环境变化所带来的挑战,通过数字化转型实现提效增益&#xff…

解决Unity-2020 安卓异形屏黑边

背景 Unity 2020.3.17 版本开发的游戏,打apk包,发现两个问题 如图下午所示,实体白色导航栏,阻挡了整个安卓UI界面,难看还影响美观。 安卓系统 12-13 版本手机,异形屏。一侧安全区黑边遮挡,占空间…

第2讲:pixi.js 绘制HelloWorld

基于第0讲和第1讲,我们增添了vite.config.ts文件。并配置了其他的http端口。 此时,我们删除掉没用的东西。 删除 conter.ts、typescript.svg 在main.ts中改成如下内容: import {Application, Text} from pixi.js import ./style.css// 指明…

数组元素的内存地址计算【数据结构与算法C#版】

数组元素被存储在连续的内存空间中,这意味着计算数组元素的内存地址非常容易。给定数组内存地址(首 元素内存地址)和某个元素的索引,我们可以使用下方图 所示的公式计算得到该元素的内存地址,从而直接 访问该元素。 观…

C# Winform图形绘制

WinForms 应用程序中的控件是基于窗体的,当控件需要重绘时,它会向父窗体发送一个消息请求重绘。但是,控件本身并不直接处理绘制命令,所以你不能直接在控件上绘制图形。 解决方法: 重写控件的OnPaint方法使用CreateGr…

五大维度大比拼:ChatGPT比较文心一言,你的AI助手选择指南

文章目录 一、评估AI助手的五个关键维度二、ChatGPT和文心一言的比较 评估AI助手的五个关键维度,以及ChatGPT和文心一言的比较如下: 一、评估AI助手的五个关键维度 界面友好性 : 评估标准:用户界面是否直观易用,是否…

Java基础 - 练习(一)打印等腰三角形

Java基础练习 打印等腰三角形,先上代码: public static void main(String[] args) {// 打印等腰三角形System.out.println("打印等腰三角形:");isoscelesTriangle(); } public static void isoscelesTriangle() {// for循环控制行…

探索未来工作新伙伴:机器人流程自动化(RPA)揭秘

想象一下,如果你的日常工作中那些繁琐、重复的任务,比如数据录入、文件整理、邮件发送等,都能自动完成,你将拥有更多时间专注于真正需要创造力和智慧的工作,是不是听起来就像拥有了一个私人助理?这并不是遥…

数据结构之线性表(3)

数据结构之线性表(3) 上文我们了解了线性表的静动态存储的相关操作,此篇我们对线性表中链表的相关操作探讨。 在进行链表的相关操作时,我们先来理解单链表是什么? 1.链表的概念及结构 链表是一种物理存储结构上非连…

C++命名空间

命名空间(namespace)的目的&#xff1a;对标识符的名称进行本地化&#xff0c;以避免命名冲突或者名字污染 #include<stdio.h> #include<stdlib.h>int rand 0; int main() { //C语言没有办法解决类似这种的命名冲突&#xff0c;而C提出了namespace来解决printf(&…

【日常记录】【vue】vite-plugin-inspect 插件的使用

文章目录 1、vite-plugin-inspect2、安装3、使用4、链接 1、vite-plugin-inspect vite-plugin-inspect 可以让开发者在浏览器端就可以看到vue文件编译后的代码、vue文件的相互依赖关系 2、安装 npm i -D vite-plugin-inspect// vite.config.ts import Inspect from vite-plugi…