深入理解 NumPy 数组的交换(Swapping)的实现
一、数组交换的核心定义
数组交换本质是调整数组中元素或维度的位置,主要分为两类场景:
- 元素级交换:交换数组中指定位置的单个 / 多个元素(如交换索引 0 和索引 5 的元素);
- 维度级交换:交换数组的行 / 列、或调整维度的整体顺序(如交换二维数组的第 1 列和第 3 列、交换三维数组的维度 0 和维度 2)。
NumPy 实现交换的核心优势是向量化操作,无需循环,直接通过索引 / 切片即可高效完成。
二、基础交换:单个 / 多个元素交换
这是最直观的交换场景,核心逻辑是「通过索引直接赋值」,利用 NumPy 支持的多元素同时赋值特性(无需临时变量)。
示例 1:一维数组交换单个元素
import numpy as np
# 一维数组(模拟10个学生分数)
scores = np.array([85, 92, 78, 90, 88, 76, 95, 81, 89, 93])
print("原始数组:", scores)
# 交换索引1(92)和索引5(76)的元素(无需临时变量)
scores[1], scores[5] = scores[5], scores[1]
print("交换后数组:", scores) # [85 76 78 90 88 92 95 81 89 93]示例 2:一维数组交换多个连续元素(切片)
import numpy as np
arr = np.array([1,2,3,4,5,6,7,8])
print("原始数组:", arr)
# 交换前3个元素(0:3)和后3个元素(5:8)
arr[0:3], arr[5:8] = arr[5:8].copy(), arr[0:3].copy()
# 注意:必须加.copy(),否则切片是视图,赋值会相互覆盖
print("交换后数组:", arr) # [6 7 8 4 5 1 2 3]关键说明:切片交换时,若直接用 arr[0:3], arr[5:8] = arr[5:8], arr[0:3],由于切片是视图,第一步赋值会修改原数组,导致第二步赋值拿到的是已修改的数据,最终结果错误。必须用 .copy() 生成独立拷贝。
示例 3:二维数组交换单个元素
import numpy as np
# 二维数组(3天×4小时温度)
temp = np.array([
[5.2, 6.1, 7.3, 8.2],
[4.8, 5.5, 6.7, 7.9],
[6.3, 7.2, 8.1, 9.0]
])
print("原始数组:\n", temp)
# 交换(0,1)(第1天第2小时)和(2,3)(第3天第4小时)的元素
temp[0,1], temp[2,3] = temp[2,3], temp[0,1]
print("交换后数组:\n", temp)输出:
原始数组:
[[5.2 6.1 7.3 8.2]
[4.8 5.5 6.7 7.9]
[6.3 7.2 8.1 9.0]]
交换后数组:
[[5.2 9.0 7.3 8.2]
[4.8 5.5 6.7 7.9]
[6.3 7.2 8.1 6.1]]
三、核心交换:二维数组的行 / 列交换
这是日常使用中最高频的交换场景,核心是「通过行 / 列索引切片实现整行 / 整列交换」。
1. 交换两行(整行)
语法:arr[行索引1], arr[行索引2] = arr[行索引2], arr[行索引1]
import numpy as np
# 二维数组(4行3列)
arr = np.array([
[1, 2, 3],
[4, 5, 6],
[7, 8, 9],
[10, 11, 12]
])
print("原始数组:\n", arr)
# 交换第1行(索引0)和第3行(索引2)
arr[0], arr[2] = arr[2].copy(), arr[0].copy()
# 加.copy()更稳妥,避免视图覆盖(低版本NumPy可能有问题)
print("交换第1/3行后:\n", arr)输出:
原始数组:
[[ 1 2 3]
[ 4 5 6]
[ 7 8 9]
[10 11 12]]
交换第1/3行后:
[[ 7 8 9]
[ 4 5 6]
[ 1 2 3]
[10 11 12]]
2. 交换两列(整列)
语法:arr[:, 列索引1], arr[:, 列索引2] = arr[:, 列索引2], arr[:, 列索引1]
import numpy as np
arr = np.array([
[1, 2, 3],
[4, 5, 6],
[7, 8, 9],
[10, 11, 12]
])
print("原始数组:\n", arr)
# 交换第2列(索引1)和第3列(索引2)
arr[:, 1], arr[:, 2] = arr[:, 2].copy(), arr[:, 1].copy()
print("交换第2/3列后:\n", arr)输出:
原始数组:
[[ 1 2 3]
[ 4 5 6]
[ 7 8 9]
[10 11 12]]
交换第2/3列后:
[[ 1 3 2]
[ 4 6 5]
[ 7 9 8]
[10 12 11]]
3. 交换多行 / 多列(批量)
import numpy as np
arr = np.arange(16).reshape(4,4)
print("原始数组:\n", arr)
# 交换前两行(0:2)和后两行(2:4)
arr[0:2], arr[2:4] = arr[2:4].copy(), arr[0:2].copy()
print("交换前两行/后两行:\n", arr)
# 重置数组
arr = np.arange(16).reshape(4,4)
# 交换前两列(:,0:2)和后两列(:,2:4)
arr[:, 0:2], arr[:, 2:4] = arr[:, 2:4].copy(), arr[:, 0:2].copy()
print("交换前两列/后两列:\n", arr)四、进阶交换:维度(轴)交换
高维数组(三维 / 四维)的交换核心是「调整维度的整体顺序」,主要通过 np.swapaxes() 和 arr.transpose() 实现(swapaxes 更专注于 “交换两个维度”)。
1. 核心函数:np.swapaxes ()
语法:np.swapaxes(arr, axis1, axis2) 或 arr.swapaxes(axis1, axis2)
axis1/axis2:要交换的两个维度编号;- 返回值:维度交换后的新数组(视图,共享内存)。
示例 1:二维数组交换维度(等价于转置)
二维数组的维度 0 = 行、维度 1 = 列,交换维度 0 和 1 等价于转置:
import numpy as np
arr = np.array([[1,2,3],[4,5,6]])
print("原始数组:\n", arr)
print("原始形状:", arr.shape) # (2,3)
# 交换维度0和1
arr_swap = arr.swapaxes(0, 1)
print("交换维度后:\n", arr_swap)
print("交换后形状:", arr_swap.shape) # (3,2)(等价于arr.T)示例 2:三维数组交换维度
import numpy as np
# 三维数组:形状(2,3,4) → 维度0=2,维度1=3,维度2=4
arr_3d = np.arange(24).reshape(2,3,4)
print("原始三维数组形状:", arr_3d.shape) # (2,3,4)
# 交换维度0和维度2
arr_swap = arr_3d.swapaxes(0, 2)
print("交换维度0/2后形状:", arr_swap.shape) # (4,3,2)
# 交换维度1和维度2
arr_swap2 = arr_3d.swapaxes(1, 2)
print("交换维度1/2后形状:", arr_swap2.shape) # (2,4,3)2. 用 transpose 实现多维度交换
transpose 可同时调整多个维度的顺序,比 swapaxes 更灵活(swapaxes 仅能交换两个维度):
import numpy as np
arr_3d = np.arange(24).reshape(2,3,4)
print("原始形状:", arr_3d.shape) # (2,3,4)
# 交换维度顺序为 (2,0,1) → 等价于先交换0和2,再交换1和2
arr_trans = arr_3d.transpose(2,0,1)
print("transpose调整后形状:", arr_trans.shape) # (4,2,3)五、实战场景:交换的典型应用
场景 1:调整图像数据维度顺序
图像数据常见维度顺序:(高度, 宽度, 通道)(HWC) ↔ (通道, 高度, 宽度)(CHW),可通过维度交换实现:
import numpy as np
# 模拟图像数据:2张图片 × 32高度 × 32宽度 × 3通道(HWC)
img_hwc = np.random.randint(0,255, (2,32,32,3))
print("HWC格式形状:", img_hwc.shape) # (2,32,32,3)
# 交换维度1(高度)和维度3(通道)?→ 更简单的方式:transpose
img_chw = img_hwc.transpose(0,3,1,2)
print("CHW格式形状:", img_chw.shape) # (2,3,32,32)
# 再交换回来(CHW → HWC)
img_hwc2 = img_chw.transpose(0,2,3,1)
print("还原后形状:", img_hwc2.shape) # (2,32,32,3)场景 2:打乱数据集的行顺序(随机交换)
import numpy as np
# 模拟数据集:100个样本 × 5个特征
data = np.random.rand(100,5)
# 生成随机索引(打乱行顺序)
random_idx = np.random.permutation(100)
# 通过索引交换实现行打乱(等价于随机重排)
data_shuffled = data[random_idx]
print("原始数据前3行:\n", data[:3])
print("打乱后数据前3行:\n", data_shuffled[:3])场景 3:修正温度数据的列顺序
import numpy as np
# 模拟温度数据:31天 × 4列(凌晨、上午、下午、晚上),但列顺序错误(凌晨和晚上颠倒)
temp = np.random.normal(5,6,(31,4)).round(1)
print("原始列顺序:凌晨、上午、下午、晚上 → 实际是:晚上、上午、下午、凌晨")
# 交换第0列(晚上)和第3列(凌晨)
temp[:, 0], temp[:, 3] = temp[:, 3].copy(), temp[:, 0].copy()
print("修正后温度数据前3行:\n", temp[:3])六、避坑点与最佳实践
1. 核心避坑点
- ❌ 切片交换未加
.copy():直接用arr[0:2], arr[2:4] = arr[2:4], arr[0:2]会导致数据覆盖,必须加.copy(); - ❌ 混淆维度编号:高维数组交换维度时,记错维度 0/1/2 的含义(如把图像的通道维度编号记错);
- ❌ 认为
swapaxes返回新数组:实际是视图(共享内存),修改会影响原数组,需独立则加.copy(); - ❌ 交换多行 / 列时索引错误:如
arr[0,2]是单个元素,而非第 0 和第 2 行(正确写法是arr[[0,2]])。
2. 最佳实践
- ✅ 单个元素交换:直接
arr[i], arr[j] = arr[j], arr[i](无需 copy); - ✅ 切片 / 整行 / 整列交换:必须加
.copy(),避免视图覆盖; - ✅ 维度交换:二维数组用
arr.T或swapaxes(0,1),高维数组用swapaxes(交换两个维度)或transpose(调整多维度); - ✅ 随机交换行:用
np.random.permutation生成随机索引,通过索引重排(高效且避免手动交换); - ✅ 验证交换结果:通过
shape检查维度,通过索引取值验证元素位置是否正确。
总结
- NumPy 数组交换分为「元素级」和「维度级」两类:
- 元素级:通过索引 / 切片赋值实现,切片交换需加
.copy(); - 维度级:整行 / 列交换用索引切片,高维维度交换用
swapaxes/transpose;
- 元素级:通过索引 / 切片赋值实现,切片交换需加
- 核心语法:
- 单个元素:
arr[i], arr[j] = arr[j], arr[i]; - 整行交换:
arr[row1], arr[row2] = arr[row2].copy(), arr[row1].copy(); - 整列交换:
arr[:, col1], arr[:, col2] = arr[:, col2].copy(), arr[:, col1].copy(); - 维度交换:
arr.swapaxes(axis1, axis2);
- 单个元素:
- 避坑关键:切片交换必须加
.copy(),维度交换牢记维度编号,验证交换结果; - 常用场景:调整数据列顺序、图像维度转换、打乱数据集、修正数据行 / 列位置。
到此这篇关于深入理解 NumPy 数组的交换(Swapping)的实现的文章就介绍到这了,更多相关NumPy 数组交换内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
相关文章
python3列表删除大量重复元素remove()方法的问题详解
这篇文章主要给大家介绍了关于python3列表删除大量重复元素remove()方法的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧2021-01-01
Python np.random.randint()参数的使用及说明
Python的np.random.randint()方法可以生成指定范围内的随机整数,可生成一维或多维数组,该方法有low、high、size、dtype等参数,可根据需求灵活使用2026-04-04


最新评论