NumPy的深浅拷贝的区别与选择

简介: 关于深浅拷贝如何进行选择, 我们需要根据实际的项目需求进行使用,选择正确的拷贝方式不但能够节省内存空间,而且在效率上也会大大提高,这也是我们在项目优化方面需要进行考虑的。

大家好,我们知道中华上下五千年。习武之人不在少数。但是他们习武的类型也大概分成两种:一种是外功,主要学习手法,腿法等的力量和协调性;另一种是内功,主要锻炼的是气力,骨骼等内在的改变。

在学习Numpy库中的副本和视图的时候,接触到了其中有一个数据拷贝的选择,它跟习武的方式类似,分为深拷贝和浅拷贝。那么究竟他们有什么区别?如何进行选择?本文将带领我们一探究竟!

知道

首先,在研究深浅拷贝之前,得知道相关的一些概念这样,后面我们才能游刃有余。比如我们在看本文内容之前,你是否学习了Python基础,NumPy环境是否已经安装完成。🎯


好了,现在正式进入我们的内容。

什么是深拷贝?

深拷贝指在拷贝原数组的时候,会在内存中重新创建一个新的数组并且进行赋值,新数组中的元素进行任何操作的时候,是不会影响到原数组的,他们在数组中的内存id是不同的


什么是浅拷贝?

浅拷贝指在拷贝原数组的时候,它只不过是原数组的一个引用,并不会再内存中创建一个新的数组,所以在操作新数组中的元素的时候,原数组也是会跟着一起变化的,会影响到原数组的数组结构。


既然我们已经知道了什么是深拷贝,什么是浅拷贝。那么光凭理论是说不过去的,我们得进行实践。记得有一句话说的:“实践是检验真理的唯一途径”。那么我们开始吧!🧐

验证

首先我们得保证安装了Pyhton的编辑器,我用的是PyCharm,没有的可以在官网自行下载安装。


这里Numpy库给我们提供的深拷贝的函数是copy()函数,我们就使用这个函数进行验证。


先在编辑器中创建一个数组,并进行数组的深拷贝,查看结果如何?

"""
@Created on : 2024/6/17 11:16
@creator : er_nao
@File :numpy_28_副本和视图.py
@Description :
"""
import numpy as np
# copy()函数的使用
x = np.arange(32).reshape(4, 8)
print('原数组')
print(x)
# 获取数组在内存中的id
print('x数组的id: ')
print(id(x))
print('\n')
# 将x数组copy给y
y = x.copy()
print('y数组的id:')
print(id(y))
print('\n')
# 修改y数组,测试x数组是否会发生变化
y.shape = 8, 4
print('打印y数组')
print(y)
print('\n')
print('打印x数组')
print(x)
# # 输出结果
# 原数组
# [[ 0  1  2  3  4  5  6  7]
#  [ 8  9 10 11 12 13 14 15]
#  [16 17 18 19 20 21 22 23]
#  [24 25 26 27 28 29 30 31]]
# x数组的id: 
# 1972961081072
# 
# 
# y数组的id:
# 1973520214672
# 
# 
# 打印y数组
# [[ 0  1  2  3]
#  [ 4  5  6  7]
#  [ 8  9 10 11]
#  [12 13 14 15]
#  [16 17 18 19]
#  [20 21 22 23]
#  [24 25 26 27]
#  [28 29 30 31]]
# 
# 
# 打印x数组
# [[ 0  1  2  3  4  5  6  7]
#  [ 8  9 10 11 12 13 14 15]
#  [16 17 18 19 20 21 22 23]
#  [24 25 26 27 28 29 30 31]]


我们可以看到,创建了一个原数组x,然后将x数组深拷贝给新数组y,输出的结果显而易见,x在内容中的id和y的id是不一样的,显然这儿变成了两个不同的数组,然后使用shape函数,进行y数组的变化,输出结果是y数组格式已经变化,而原数组并没有收到影响。


下来我们验证浅拷贝,浅拷贝其实就比较简单了,就是数组之间的赋值。我们来看一下相关代码

# 传统数组之间相互赋值
x = np.array([1, 2, 3, 4, 5, 6, 7, 8])
print('原数组')
print(x)
# 获取数组在内存中的id
print('x数组的id: ')
print(id(x))
print('\n')
# 将x数组赋值给y
y = x
print('y数组的id:')
print(id(y))
print('\n')
# 修改y数组,测试x数组是否会发生变化
y.shape = 2, 4
print('打印y数组')
print(y)
print('\n')
print('打印x数组')
print(x)
# 输出结果
# 原数组
# [1 2 3 4 5 6 7 8]
# x数组的id: 
# 1795409508176
# 
# 
# y数组的id:
# 1795409508176
# 
# 
# 打印y数组
# [[1 2 3 4]
#  [5 6 7 8]]
# 
# 
# 打印x数组
# [[1 2 3 4]
#  [5 6 7 8]]


这次我们可以很清晰的看到,将x数组赋值给数组y,改变数组y以后,x也改变了。并且他们的内存id都是一样的。


其实NumPy库还给我们提供了一种浅拷贝的方法, 使用view()函数进行拷贝。

# 创建原始数组
arr = np.array([2, 4, 8, 12, 20])
# 使用view函数创建视图
view_arr = arr.view()
# 修改原始数组中的元素
arr[0] = 100
# 输出修改后的结果
print("原数组id:", id(arr))
print("视图数组id:", id(view_arr))
print("原数组:", arr)
print("视图数组:", view_arr)
# 输出结果
# 原数组id: 1958732690896
# 视图数组id: 1958732691280
# 原数组: [100   4   8  12  20]
# 视图数组: [100   4   8  12  20]


我们发现,view()函数的是创建一个新的视图,但是它在元素内存的指向和原数组的相同的,改变原数值的元素,创建的视图元素也随之改变。

清楚

以上就是本文的全部内容,关于深浅拷贝如何进行选择, 我们需要根据实际的项目需求进行使用,选择正确的拷贝方式不但能够节省内存空间,而且在效率上也会大大提高,这也是我们在项目优化方面需要进行考虑的。

相关文章
|
Python
【Numpy】深入剖析Numpy.arange()与range()的区别
【Numpy】深入剖析Numpy.arange()与range()的区别
370 0
|
Python
numpy数组中冒号[:,:,0]与[...,0]的区别
x[:,:,0] 意思是对数组x切片,可以想象成一个正方体数据,每次切下一个面的数据。第二维取0则得出来[3,4]大小的数组
434 0
|
机器学习/深度学习 存储 Serverless
NumPy 与 Python 内置列表计算标准差的区别
NumPy,是 Numerical Python 的简称,用于高性能科学计算和数据分析的基础包,像数学科学工具(pandas)和框架(Scikit-learn)中都使用到了 NumPy 这个包。
|
Python
numpy中dot与*的区别
dot是矩阵相乘,只要矩阵满足前一个矩阵的列与后一个矩阵的行相匹配即可 *是遵循numpy中的广播机制,必须保证行列相对应才可以进行运算 先看一则正例 >>import numpy as np #test1与test2行列相同 >>test1 = np.
1163 0
|
2月前
|
存储 Java 数据处理
(numpy)Python做数据处理必备框架!(一):认识numpy;从概念层面开始学习ndarray数组:形状、数组转置、数值范围、矩阵...
Numpy是什么? numpy是Python中科学计算的基础包。 它是一个Python库,提供多维数组对象、各种派生对象(例如掩码数组和矩阵)以及用于对数组进行快速操作的各种方法,包括数学、逻辑、形状操作、排序、选择、I/0 、离散傅里叶变换、基本线性代数、基本统计运算、随机模拟等等。 Numpy能做什么? numpy的部分功能如下: ndarray,一个具有矢量算术运算和复杂广播能力的快速且节省空间的多维数组 用于对整组数据进行快速运算的标准数学函数(无需编写循环)。 用于读写磁盘数据的工具以及用于操作内存映射文件的工具。 线性代数、随机数生成以及傅里叶变换功能。 用于集成由C、C++
347 0
|
2月前
|
Java 数据处理 索引
(numpy)Python做数据处理必备框架!(二):ndarray切片的使用与运算;常见的ndarray函数:平方根、正余弦、自然对数、指数、幂等运算;统计函数:方差、均值、极差;比较函数...
ndarray切片 索引从0开始 索引/切片类型 描述/用法 基本索引 通过整数索引直接访问元素。 行/列切片 使用冒号:切片语法选择行或列的子集 连续切片 从起始索引到结束索引按步长切片 使用slice函数 通过slice(start,stop,strp)定义切片规则 布尔索引 通过布尔条件筛选满足条件的元素。支持逻辑运算符 &、|。
187 0
|
4月前
|
机器学习/深度学习 API 异构计算
JAX快速上手:从NumPy到GPU加速的Python高性能计算库入门教程
JAX是Google开发的高性能数值计算库,旨在解决NumPy在现代计算需求下的局限性。它不仅兼容NumPy的API,还引入了自动微分、GPU/TPU加速和即时编译(JIT)等关键功能,显著提升了计算效率。JAX适用于机器学习、科学模拟等需要大规模计算和梯度优化的场景,为Python在高性能计算领域开辟了新路径。
416 0
JAX快速上手:从NumPy到GPU加速的Python高性能计算库入门教程
|
4月前
|
存储 数据采集 数据处理
Pandas与NumPy:Python数据处理的双剑合璧
Pandas与NumPy是Python数据科学的核心工具。NumPy以高效的多维数组支持数值计算,适用于大规模矩阵运算;Pandas则提供灵活的DataFrame结构,擅长处理表格型数据与缺失值。二者在性能与功能上各具优势,协同构建现代数据分析的技术基石。
404 0
|
机器学习/深度学习 数据处理 Python
从NumPy到Pandas:轻松转换Python数值库与数据处理利器
从NumPy到Pandas:轻松转换Python数值库与数据处理利器
337 1
|
机器学习/深度学习 数据处理 计算机视觉
NumPy实践宝典:Python高手教你如何轻松玩转数据处理!
【8月更文挑战第22天】NumPy是Python科学计算的核心库,专长于大型数组与矩阵运算,并提供了丰富的数学函数。首先需安装NumPy (`pip install numpy`)。之后可通过创建数组、索引与切片、执行数学与逻辑运算、变换数组形状及类型、计算统计量和进行矩阵运算等操作来实践学习。NumPy的应用范围广泛,从基础的数据处理到图像处理都能胜任,是数据科学领域的必备工具。
195 0