numpy-常用函数总结

numpy-常用函数总结

学习地址:https://blog.csdn.net/zkx990121/article/details/119136515

1. Axis理解

1.1 Axis维度

简单来说,最外面的括号代表着axis=0,依次往里的括号对应的axis的计数就依次加1

操作方式:如果指定轴进行相关的操作,那么会使用轴下的每一个直接子元素的第0个,第1个,第2个…分别进行相关的操作示例:

x = np.array([[0,1],[2,3]])
print(x)
# [[0 1]
#  [2 3]]

1.求 x 数组在axis = 0 和 axis=1 两种情况下的和

# 求列和
x.sum(axis=0)
print("x.sum(axis=0):", x.sum(axis=0))
# x.sum(axis=0): [2 4]

# 求行和
x.sum(axis=1)
print("x.sum(axis=1):", x.sum(axis=1))
# x.sum(axis=1): [1 5]

2.用np.max求 axis=0 和 axis=1 两种情况下的最大值

np.random.seed(100)
x = np.random.randint(1,10,size=(3,5))
print("np.random.randint(1,10,size=(3,5)) x: \n", x)
# np.random.randint(1,10,size=(3,5)) x:
# [[9 9 4 8 8]
#  [1 5 3 6 3]
#  [3 3 2 1 9]]

x.max(axis=0)
print("x.max(axis=0): \n", x.max(axis=0))
# x.max(axis=0):
#  [9 9 4 8 9]

x.max(axis=1)
print("x.max(axis=1): \n", x.max(axis=1))
# x.max(axis=1):
#  [9 6 9]

3.用 np.deleteaxis=0axis=1 两种情况下删除元素

np.random.seed(100)
x = np.random.randint(1,10,size=(3,5))
print("np.random.randint(1,10,size=(3,5)): \n", x)
# np.random.randint(1,10,size=(3,5)) x: 
#  [[9 9 4 8 8]
#  [1 5 3 6 3]
#  [3 3 2 1 9]]

#删除第0行
np.delete(x,0,axis=0)
print("np.delete(x,0,axis=0): \n", np.delete(x,0,axis=0))
# np.delete(x,0,axis=0):
#  [[1 5 3 6 3]
#  [3 3 2 1 9]]

#删除第0列
np.delete(x,0,axis=1)
print("np.delete(x,0,axis=1): \n", np.delete(x,0,axis=1))
# np.delete(x,0,axis=1):
#  [[9 4 8 8]
#  [5 3 6 3]
#  [3 2 1 9]]

1.2 三维及多维数组

#生成一个三维数组
y = np.arange(24).reshape(2,2,6)
print("生成一个三维数组 np.arange(24).reshape(2,2,6): \n", 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]]]

#取最大值
y.max(axis=0)
print("y.max(axis=2): \n", y.max(axis=2))
#  [[ 5 11]
#  [17 23]]

1.3 总结

  1. 最外面的括号代表着 axis=0,依次往里的括号对应的 axis 的计数就依次加1
  2. 如果指定轴进行相关的操作,那么他会使用轴下面的每个子元素的第0个,第1个,第2个...分别进行相关的操作
  3. np.delete是直接删除指定轴下的第几个直接子元素

2. 创建数组

2.1 使用np.array创建

q = np.array([1, 2, 3, 4])
print(q)      
# [1 2 3 4]

print(type(q)) 
# <class 'numpy.ndarray'>

2.2 使用np.arange创建

#创建0-10步数为2的数组 结果为[0,2,4,6,8]
b = np.arange(0, 10, 2)
print(b) 
# [0 2 4 6 8]

2.3 np.random.random

#3. np.random.random创建一个N行N列的数组
# 其中里面的值是0-1之间的随机数
# 创建3行3列的数组
c = np.random.random((3, 3))
print(c)
# [[ 0.28709175  0.0133334   0.9399995 ]
#  [ 0.81712176  0.43983614  0.69228111]
#  [ 0.75798902  0.2459419   0.84720639]]

2.4 np.random.randint

#4. np.random.randint创建一个N行N列的数组
# 其中值的范围可以通过前面2个参数来指定
# 创建值的范围为[0,9)的5行5列数组
d = np.random.randint(0, 10, size=(5, 5))
print(d)
# [[4 5 6 9 1]
#  [8 0 4 0 7]
#  [3 8 3 9 6]
#  [2 9 4 4 9]
#  [3 1 9 9 1]]

2.5 特殊函数

2.5.1 np.zeros

## N行N列的全零数组
### 例如:3行3列全零数组
array_zeors = np.zeros((3, 3))
print(array_zeors)
# [[ 0.  0.  0.]
#  [ 0.  0.  0.]
#  [ 0.  0.  0.]]

2.5.2 np.ones

## N行N列的全一数组
### 例如:5行5列全一数组
array_ones = np.ones((5, 5))
print(array_ones)
# [[ 1.  1.  1.  1.  1.]
#  [ 1.  1.  1.  1.  1.]
#  [ 1.  1.  1.  1.  1.]
#  [ 1.  1.  1.  1.  1.]
#  [ 1.  1.  1.  1.  1.]]

2.5.3 np.full

## 全部为指定值的N行N列数组
### 例如:值为0的2行3列数组
array_full = np.full((3, 4), 3)
print(array_full)
# [[3 3 3 3]
#  [3 3 3 3]
#  [3 3 3 3]]

2.5.4 np.eye

## 生成一个在斜方形上元素为1,其他元素都为0的N行N列矩阵
### 例如:6行6列矩阵
array_eye = np.eye(6)
print(array_eye)
# [[ 1.  0.  0.  0.  0.  0.]
#  [ 0.  1.  0.  0.  0.  0.]
#  [ 0.  0.  1.  0.  0.  0.]
#  [ 0.  0.  0.  1.  0.  0.]
#  [ 0.  0.  0.  0.  1.  0.]
#  [ 0.  0.  0.  0.  0.  1.]]

2.6. 注意

  1. 数组中的数据类型必须一致,要么全部为整型,要么全部为浮点类型,要么全部为字符串类型
  2. 不能同时出现多种数据类型

3. random模块

3.1 np.random.seed

用于指定随机数生成时所用算法开始的整数值,如果使用相同的seed()值,则每次生成的随机数都相同,

如果不设置这个值,则系统根据时间来自己选择这个值,此时每次生成的随机数因时间差异不同。一般没有特殊要求不用设置。

3.2 np.random.rand

生成一个值为 [0,1) 之间的数组,形状由参数指定,如果没有参数,那么将返回一个随机值

#产生随机数
np.random.rand()
print("np.random.rand(): \n", np.random.rand())
# 0.0923385947687978

#产生随机数组 两行三列
np.random.rand(2,3)
print("np.random.rand(2,3): \n", np.random.rand(2,3))
# [[0.20445225 0.87811744 0.02738759]
#  [0.67046751 0.4173048  0.55868983]]

3.3 np.random.randn

生成均值(μ)为0,标准差(σ)为1的标准正态分布的值

#生成一个2行3列的数组,数组中的值都满足标准正态分布
data12 = np.random.randn(2,3)
print("np.random.randn(2,3): \n", data12)
# [[-0.3224172  -0.38405435  1.13376944]
#  [-1.09989127 -0.17242821 -0.87785842]]

3.4 np.random.randint

生成指定范围内的随机数,并且可以通过size参数指定维度

#生成值在0-10之间,3行5列的数组
data12_1 = np.random.randint(10,size=(3,5))
print("生成值在0-10之间,3行5列的数组: \n", data12_1)
# [[5 1 9 3 4]
#  [8 1 4 0 3]
#  [9 2 0 4 9]]

#生成值在1-20之间,3行6列的数组
data12_2 = np.random.randint(1,20,size=(3,6))
print("生成值在1-20之间,3行6列的数组: \n", data12_2)
# [[16  3  8  9 10  4]
#  [ 8  5  6  7  9  1]
#  [ 3 11 16 16  8 11]]

3.5 np.random.choice

从一个列表或者数组中,随机进行采样。或者是从指定的区间中进行采样,采样个数可以通过参数

#从数组中随机选择三个值
data12_3 = np.random.choice(data12.flatten(), 3)
print("从数组中随机选择三个值: \n", data12_3)
#  [-0.3224172  -0.3224172  -0.38405435]
#从数组中获取值组成新的数组
data12_4 = np.random.choice(data12.flatten(),size=(3,4))
print("从数组中获取值组成新的数组: \n", data12_4)
# [[-0.38405435 -0.87785842 -1.09989127 -0.3224172 ]
#  [-0.3224172  -0.87785842 -0.87785842 -0.17242821]
#  [-0.87785842  1.13376944 -0.17242821 -1.09989127]]

#从指定值随机取值 (示例:从0-10之间随机取3个值)
data12_5 = np.random.choice(10,3)
print("从指定值随机取值 (示例:从0-10之间随机取3个值): \n", data12_5)
#  [4 4 7]

3.6 np.random.shuffle

把原来数组的元素的位置打乱

q12 = np.arange(10)
print("np.arange(10): \n", q12)
#  [0 1 2 3 4 5 6 7 8 9]

#将数组q12的元素的位置都会进行随机更换
#shuffle没有返回值,直接打乱原数组位置
np.random.shuffle(q12)
print("shuffle没有返回值,直接打乱原数组位置: \n", q12)
#  [6 2 3 0 1 8 5 7 9 4]

4. 数组数据类型

4.1 数据类型

数据类型 描述 唯一标识符
bool 用一个字节存储的布尔类型(True或False) b
int8 一个字节大小,-128 至 127 i1
int16 整数,16 位整数(-32768 ~ 32767) i2
int32 整数,32 位整数(-2147483648 ~ 2147483647) i4
int64 整数,64 位整数(-9223372036854775808 ~ 9223372036854775807) i8
uint8 无符号整数,0 至 255 u1
uint16 无符号整数,0 至 65535 u2
uint32 无符号整数,0 至 2 ** 32 - 1 u4
uint64 无符号整数,0 至 2 ** 64 - 1 u8
float16 半精度浮点数:16位,正负号1位,指数5位,精度10位 f2
float32 单精度浮点数:32位,正负号1位,指数8位,精度23位 f4
float64 单精度浮点数:64位,正负号1位,指数11位,精度52位 f8
complex64 复数,分别用两个32位浮点数表示实部和虚部 c8
complex128 复数,分别用两个64位浮点数表示实部和虚部 c16
object_ python对象 O
string_ 字符串 S
unicode_ unicode类型 U

4.2 创建数组指定类型

# i1 是int8
q2 = np.array([1, 2, 3, 4, 5], dtype='i1')
print(q2)   # [1 2 3 4 5]
q21 = np.array([1, 2, 3, 4, 5], dtype=int)
print(q21) # [1 2 3 4 5]

4.3 查询数据类型

y = np.arange(24).reshape(2,2,6)
print(y.dtype)  #  int32

4.4 修改数据类型

c2 = np.array([1, 2, 3], dtype='i2')
print(c2.dtype) # int16

d2 = c2.astype('f2')
print(d2)       # [ 1.  2.  3.]
print(d2.dtype) # float16

4.5 总结

  1. Numpy是基于C语言编写,引用了C语言的数据类型,所以Numpy的数组中数据类型多样
  2. 不同的数据类型有利于处理海量数据,针对不同数据赋予不同数据类型,从而节省内存空间

5. 多维数组

5.1 数组维度查询

# 打印维度
a3 = np.array([1, 2, 3])
print(a3.ndim)
# 1

a32 = np.array([
    [5, 6, 7],
    [22, 33, 55]
])
print(a32.ndim)
# 2

q33 = np.array([
    [
        [33, 44, 55],
        [66, 77, 88]
    ],
    [
        [2, 1, 3],
        [1, 2, 1]
    ]
])
print(q33.ndim) # 3

5.2 数组形状查询

print(q33.shape) 
# (2, 2, 3)

# 打印行列
q34 = np.array([0, 1, 22])
print(q34.shape)   # (3,)


q35 = np.array([[0, 1, 22], [2, 1, 3]])
print(q35.shape) # (2, 3)

5.3 修改数组形状

# 以 q33为例
print(q33)
# [[[33 44 55]
#   [66 77 88]]
#
#  [[ 2  1  3]
#   [ 1  2  1]]]

print(q33.shape)   # (2, 2, 3)
q33_1 = q33.reshape(3, 4)
print(q33_1)
# [[33 44 55 66]
#  [77 88  2  1]
#  [ 3  1  2  1]]
print(q33_1.shape) # (3, 4)

# 扁平化 (多维数组转化为一维数组)
q33_2 = q33.flatten()
print(q33_2) # [33 44 55 66 77 88  2  1  3  1  2  1]
print(q33.ndim) # 3

5.4 元素个数和内存

# 数组元素个数
count = q33.size
print(count)  # 12
# 各元素所占内存
print(q33.itemsize) # 4
# 各元素类型
print(q33.dtype) # int32
# 数组所占内存
print(q33.itemsize * q33.size) # 48

5.5 总结

  1. 一般情况下,数组维度最大到三维,一般会把三维以上的数组转化为二维数组来计算

  2. ndarray.ndim 查询数组的维度

  3. ndarray.shape 可以看到数组的形状(几行几列),shape是一个元组,里面有几个元素代表是几维数组

  4. ndarray.reshape 可以修改数组的形状。条件只有一个,就是修改后的形状的元素个数必须和原来的个数一致。

    reshape 不会修改原来数组的形状,只会将修改后的结果返回。

  5. ndarray.size 查询数组元素个数

  6. ndarray.itemsize 可以看到数组中每个元素所占内存的大小,单位是字节。(1个字节=8位)

6. 数组索引和切片

6.1 一维数组

# 6.1 一维数组
q41 = np.arange(10)
print("q41: ", q41)         	# q41:  [0 1 2 3 4 5 6 7 8 9]
print("q41[5]: " , q41[5])  	# q41[5]:  5
print("q41[-1]: " , q41[-1])  	# q41[-1]:  9

# 6.2 切片
print("切片: q41[2:5]: ", q41[2:5]) # [2 3 4]

# 6.3 使用步长
print("使用步长 q41[::3]: ", q41[::3]) # [0 3 6 9]

6.2 二维数组

通过中括号来索引和切片,在中括号中使用逗号进行分割.

逗号前面的是行,逗号后面的是列,如果多维数组中只有一个值,那么这个值就是行

q42 = np.random.randint(0,10, size=(4,6))
print("二维数组: \n", q42)
#  [[4 8 6 6 0 3]
#  [8 3 6 8 3 0]
#  [5 6 7 1 8 0]
#  [9 0 5 3 0 8]]

#获取第0行数据
print("获取第0行数据: ", q42[0]) # 1

#获取第1,2行数据
print("获取第1,2行数据: ", q42[1:3]) 
#  [[6 0 9 4 1 2]
#  [4 1 4 6 8 1]]

#获取多行数据 例0,2,3行数据
print("获取多行数据 例0,2,3行数据: ", q42[[0,2,3]])
# [[9 0 7 9 4 1]
#  [4 1 4 6 8 1]
#  [1 5 0 9 9 4]]

#获取第二行第一列数据
print("获取第二行第一列数据: ", q42[2,1]) # 1

#获取多个数据 例:第一行第四列、第二行第五列数据
print("获取多个数据 例:第一行第四列、第二行第五列数据: ", q42[[1,2],[4,5]])
# [1 1]

#获取多个数据 例:第一、二行的第四、五列的数据
print("获取多个数据 例:第一、二行的第四、五列的数据: ", q42[1:3,4:6])
# [[1 2]
#  [8 1]]

#获取某一列数据 例:第一列的全部数据
print("获取某一列数据 例:第一列的全部数据: ", q42[:,1])
#  [0 0 1 5]

#获取多列数据 例:第一、三列的全部数据
print("获取多列数据 例:第一、三列的全部数据: ", q42[:,[1,3]])
#  [[0 9]
#  [0 4]
#  [1 6]
#  [5 9]]

6.3 总结

  1. 如果数组是一维的,那么索引和切片就是和python的列表是一样的
  2. 如果是多维的(这里以二维为例),那么在中括号中,给两个值,两个值是通过逗号分隔的,逗号前面的是行,逗号后面的是列。如果中括号中只有一个值,那么就是代表行。
  3. 如果是多维数组(以二维为例),那么行的部分和列的部分,都是遵循一维数组的方式,可以使用整型、切片,还可以使用中括号的形式代表不连续的。比如a[[1,2],[3,4]],那么返回的就是第一行第三列、第二行第四列的两个值。

7. 布尔索引

# 1 生成1-24的4行6列的二维数组
q5 = np.arange(24).reshape((4,6))

# 2 获取比10小的元素
print("获取比10小的元素: \n", q5[q5<10])
#array([0, 1, 2, 3, 4, 5, 6, 7, 8, 9])

# 3 获取小于5或大于10的元素
ret = q5[(q5<5) | (q5>10)]
print("获取小于5或大于10的元素: ", ret)
# array([ 0,  1,  2,  3,  4, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22,23])

7.1总结

  1. 布尔索引是通过相同数据上的 True 还是 False 来进行提取的。
  2. 提取条件可以为一个或多个,当提取条件为多个时使用&代表且,使用 | 代表或
  3. 当提取条件为多个时,每个条件要使用圆括号括起来

8. 数组元素值的替换

8.1 方式一:索引

利用索引可以做值的替换,把满足条件的位置的值替换成其他值

#创建数组元素值为[0,10)随机数的3行5列数组
q6 = np.random.randint(0,10,size=(3,5))
print("创建数组元素值为[0,10)随机数的3行5列数组: \n", q6)
#  [[4 1 7 6 9]
#  [7 2 4 2 9]
#  [2 4 0 7 4]]

#将q6数组第2行数据全部更换为33
q6[1] = 33
print("将q6数组第一行数据全部更换为0: \n", q6)

#将q6数组第2行数据更换为[1,2,3,4,5] -- 数据个数要对应
q6[1] = np.array([1,2,3,4,5])
print("将q6数组第2行数据更换为[1,2,3,4,5]: \n", q6)

8.2 方式二:条件索引

#数组中值小于3的元素全部替换为1
q6[q6 < 3] = 1
print("数组中值小于3的元素全部替换为1: \n", q6)
#  [[7 8 8 6 1]
#  [1 1 3 4 5]
#  [9 3 4 8 4]]

8.3 方式三:函数

#将q6数组中小于5的值替换为2,剩余值替换为3
result = np.where(q6<5, 2, 3)
print("将q6数组中小于5的值替换为2,剩余值替换为3: \n", result)
#  [[2 3 2 2 3]
#  [2 2 2 2 3]
#  [2 3 2 2 2]]

8.4 总结

  1. 使用索引或者切片来替换值
  2. 使用条件索引来替换值
  3. 使用where函数来实现替换值

9. 数组的广播机制

9.0. 数组的广播原则

如果两个数组的后缘维度(即从末尾开始算起的维度)的轴长度相符或其中一方的长度为1,则认为他们是广播兼容的。广播会在全是和(或)长度为1的维度上进行。

9.1 数组与数字运算

#生成3行5列 值为0-5随机整数的数组
q7 = np.random.randint(0,5,size=(3,5))
print("生成3行5列 值为0-5随机整数的数组: \n", q7)
#  [[0 2 1 1 0]
#  [0 1 4 4 2]
#  [4 0 3 2 3]]

#数组中的所有元素都乘2
print("数组中的所有元素都乘2: \n", q7*2)
#  [[0 4 2 2 0]
#  [0 2 8 8 4]
#  [8 0 6 4 6]]

#数组中所有的元素只保留2位小数
print("数组中所有的元素只保留2位小数: \n", q7.round(2))
#  [[0 2 1 1 0]
#  [0 1 4 4 2]
#  [4 0 3 2 3]]

9.2 数组与数组运算

  1. 数组形状一致时 各个元素相加减(满足数组广播机制)
# q7 
#  [[3 2 2 3 4]
#  [1 3 3 1 2]
#  [3 4 1 0 2]]
q72 = np.random.randint(0,5,size=(3,5))
print("3*5 [0,5)数组: \n", q72)
#  [[0 2 0 4 0]
#  [0 4 0 3 1]
#  [2 3 2 2 2]]
print("q7 + q72: \n", q7 + q72)
#  [[3 4 2 7 4]
#  [1 7 3 4 3]
#  [5 7 3 2 4]]
  1. 形状不一致的数组不能相加减(不满足数组广播机制)
q72 = np.random.randint(0,5,size=(3,4))
# print("q7 + q72 形状不一致: \n", q7 + q72)
# q7 + q72 报错 : ValueError: operands could not be broadcast together with shapes (3,5) (3,4)
  1. 两个数组行数相同 ,其中一个数组列数为1(满足数组广播机制)
q73 = np.random.randint(0,5,size=(3,1))
# q7 
#  [[0 1 3 1 1]
#  [4 4 0 4 2]
#  [1 3 2 2 2]]
print("q73: \n", q73)
#  [[2]
#  [4]
#  [2]]
print("q7 size: {}, q73 size: {}".format(q7.shape, q73.shape))
# q7 size: (3, 5), q73 size: (3, 1)
print("q7+q73: \n", q7+q73)
#  [[2 3 5 3 3]
#  [8 8 4 8 6]
#  [3 5 4 4 4]]
  1. 两个数组列数相同 ,其中一个数组行数为1(满足数组广播机制)
q74 = np.random.randint(0,5,size=(1,5))
# q7 
#  [[0 1 3 1 1]
#  [4 4 0 4 2]
#  [1 3 2 2 2]]

print("q74: \n", q74)
#  [[2 4 1 1 1]]

print("q7 size: {}, q74 size: {}".format(q7.shape, q74.shape))
# q7 size: (3, 5), q74 size: (1, 5)

print("q7+q74: \n", q7+q74)
#  [[2 5 4 2 2]
#  [6 8 1 5 3]
#  [3 7 3 3 3]]

9.3总结

  1. 数组和数字直接进行运算是没有问题的
  2. 两个 shape 想要的数组是可以进行运算的
  3. 如果两个 shape 不同的数组,想要进行运算,那么需要看他们是否满足广播原则

10. 数组形状的操作

10.1 数组形状的改变

10.1.1 reshape 与 resize

reshape 与 resize 都是用来修改数组形状的,但是存在不同:

  • reshape 是将数组转换成指定的形状,然后返回转换后的结果,对于原数组的形状是不会发生改变的
  • resize 是将数组转换成指定的形状,会直接修改数组本身,并且不会返回任何值
q81 = np.random.randint(0,10,size=(3,4))
print("q81: \n", q81)

# reshape是将数组转换成指定的形状,然后返回转换后的结果,对于原数组的形状是不会发生改变的
q82 = q81.reshape((2,6))
print("q81: \n", q81)
#  [[2 9 2 7]
#  [9 1 6 9]
#  [0 5 7 3]]
print("q82: \n", q82)
#  [[2 9 2 7 9 1]
#  [6 9 0 5 7 3]]

# resize是将数组转换成指定的形状,会直接修改数组本身,并且不会返回任何值
q82 = q81.resize((4,3))
print("q81: \n", q81)
#  [[2 9 2]
#  [7 9 1]
#  [6 9 0]
#  [5 7 3]]
print("q82: \n", q82)
#  None

10.1.2 flatten 与 ravel

faltten 与 ravel 都是将多维数组转换为一维数组,但是存在不同:

  • flatten 是将数组转换为一维数组后,然后将这个拷贝返回回去,然后后续对这个返回值进行修改不会影响之前的数组
  • ravel 是将数组转换为一维数组后,将视图(引用)返回回去,后续对这个返回值进行修改会影响之前的数组
q83 = np.random.randint(0,10,size=(3,4))
print("q83: \n", q83)
#  [[6 3 9 4]
#  [8 1 3 7]
#  [2 2 5 4]]

# flatten是将数组转换为一维数组后,然后将这个拷贝返回回去,然后后续对这个返回值进行修改不会影响之前的数组
q84 = q83.flatten()
print("q84: \n", q84)
# [6 3 9 4 8 1 3 7 2 2 5 4]
q84[0] = 100
print("q83[0,0]: \n", q83[0,0])
# 结果为:6
print("q84[0]: \n", q84[0])
# 结果为:100

# ravel是将数组转换为一维数组后,将这个视图(引用)返回回去,后续对这个返回值进行修改会影响之前的数组
q85 = q83.ravel()
q85[0] = 100
print("q85[0]: \n", q85[0])
# 结果为:100
print(q83[0,0])
print("q83[0,0]: \n", q83[0,0])
# 结果为:100

10.2 数组的叠加

vstack代表在垂直方向叠加,如果想要叠加成功,那么列数必须一致 hstack代表在水平方向叠加,如果想要叠加成功,那么行数必须一致 concatenate 可以手动的指定axis参数具体在哪个方向叠加:

  • 如果axis=0,代表在水平方向叠加
  • 如果axis=1,代表在垂直方向叠加
  • 如果axis=None,会先进行叠加,再转化为1维数组
vstack1 = np.random.randint(0,10,size=(3,4))
print("vstack1: \n", vstack1)
#  [[0 7 6 4]
#  [2 6 9 3]
#  [9 9 1 4]]
vstack2 = np.random.randint(0,10,size=(2,4))
print("vstack2: \n", vstack2)
#  [[3 2 6 1]
#  [8 3 7 2]]

#垂直方向叠加的两种方式
vstack3 = np.vstack([vstack1,vstack2])
print("vstack3: \n", vstack3)
#  [[0 7 6 4]
#  [2 6 9 3]
#  [9 9 1 4]
#  [3 2 6 1]
#  [8 3 7 2]]
vstack4 = np.concatenate([vstack1,vstack2],axis=0)
print("vstack4: \n", vstack4)
#  [[0 7 6 4]
#  [2 6 9 3]
#  [9 9 1 4]
#  [3 2 6 1]
#  [8 3 7 2]]

h1 = np.random.randint(0,10,size=(3,4))
print("h1: \n", h1)
#  [[6 7 2 7]
#  [4 3 1 4]
#  [2 1 7 9]]
h2 = np.random.randint(0,10,size=(3,1))
print("h2: \n", h2)
#  [[9]
#  [4]
#  [5]]

#水平方向叠加的两种方式
h3 = np.hstack([h2,h1])
print("h3: \n", h3)
#  [[9 6 7 2 7]
#  [4 4 3 1 4]
#  [5 2 1 7 9]]
h4 = np.concatenate([h2,h1],axis=1)
print("h4: \n", h4)
#  [[9 6 7 2 7]
#  [4 4 3 1 4]
#  [5 2 1 7 9]]
# [9 4 5 6 7 2 7 4 3 1 4 2 1 7 9]

#先识别垂直叠加或水平叠加 后转换为一维数组
h5 = np.concatenate([h2,h1],axis=None)
print(h5)
print("h5: \n", h5)
#  [9 4 5 6 7 2 7 4 3 1 4 2 1 7 9]

10.3 数组的切割

hsplit 代表在水平方向切割,按列进行切割。

hsplit 切割方式两种:

第1种直接指定平均切割成多少列,

第2种是指定切割的下标值

vsplit 代表在垂直方向切割,按行进行切割。切割方式与 hsplit 相同

split/array_split 是手动的指定 axis 参数,axis=0 代表按行进行切割,axis=1代表按列进行切割

hs1 = np.random.randint(0,10,size=(3,4))
print("hs1: \n", hs1)
#  [[8 6 5 3]
#  [2 6 4 9]
#  [3 8 2 6]]

#水平方向平均分为2份  (要求列数可被此数整除)
np.hsplit(hs1,2)
print("np.hsplit(hs1,2): \n", np.hsplit(hs1,2))
#  [array([[8, 6],
#        [2, 6],
#        [3, 8]]),
#   array([[5, 3],
#        [4, 9],
#        [2, 6]])]

#水平方向分为1,1,2列(在下标为1,2处切割)
np.hsplit(hs1,(1,2))
print("np.hsplit(hs1,(1,2)): \n", np.hsplit(hs1,(1,2)))
# [array([[8],
#        [2],
#        [3]]),
#  array([[6],
#        [6],
#        [8]]),
#  array([[5, 3],
#        [4, 9],
#        [2, 6]])]

vs1 = np.random.randint(0,10,size=(4,5))
print("vs1: \n", vs1)
# [[3 4 6 3 1]
#  [7 9 2 9 6]
#  [8 0 2 8 0]
#  [5 9 3 2 3]]

#垂直方向平均分为4份
print("np.vsplit(vs1,4): \n", np.vsplit(vs1,4))
# [array([[3, 4, 6, 3, 1]]),
#  array([[7, 9, 2, 9, 6]]),
#  array([[8, 0, 2, 8, 0]]),
#  array([[5, 9, 3, 2, 3]])]

#垂直方向分为1,2,1
print("np.vsplit(vs1,(1,3)): \n", np.vsplit(vs1,(1, 3)))
#  [array([[3, 4, 6, 3, 1]]),
#   array([[7, 9, 2, 9, 6],
#          [8, 0, 2, 8, 0]]),
#   array([[5, 9, 3, 2, 3]])]

#split/array_split(array,indicate_or_section,axis):用于指定切割方式,在切割的时候需要指定按照行还是列,axis=1代表按照列,axis=0代表按照行
#按列平均切割
print("np.split(hs1,4,axis=1): \n", np.split(hs1,4,axis=1))
# [array([[8],
#        [2],
#        [3]]),
#  array([[6],
#        [6],
#        [8]]),
#  array([[5],
#        [4],
#        [2]]),
#  array([[3],
#        [9],
#        [6]])]

#按行平均切割
np.split(vs1,4,axis=0)
print("np.split(vs1,4,axis=0): \n", np.split(vs1,4,axis=0))
#  [array([[3, 4, 6, 3, 1]]), array([[7, 9, 2, 9, 6]]), array([[8, 0, 2, 8, 0]]), array([[5, 9, 3, 2, 3]])]

10.4 矩阵转置

ndarray.T 转置

t1 = np.random.randint(0,10,size=(3,4))
print("t1: \n", t1)
# [[4 5 6 4]
#  [2 5 4 5]
#  [6 1 0 0]]

#数组t1转置
t1.T
print("t1.T 转置: \n", t1.T)
# [[4 2 6]
#  [5 5 1]
#  [6 4 0]
#  [4 5 0]]

#矩阵相乘
t1.dot(t1.T)
print("t矩阵相乘 t1.dot(t1.T): \n", t1.dot(t1.T))
# [[93 77 29]
#  [77 70 17]
#  [29 17 37]]

ndarray.transpose() 转置

#transpose返回的是一个View,所以对返回值上进行修改会影响到原来的数组。
t2 = t1.transpose()
print("transpose()转置: \n", t2)
# [[4 2 6]
#  [5 5 1]
#  [6 4 0]
#  [4 5 0]]

10.5 总结

  1. 数据的形状改变

    (1)reshape和resize 都是重新定义形状的,但是 reshape 不会修改数组本身,而是将修改后的结果返回回去,而 resize 是直接修改数组本身的

    (2) flatten 和 ravel 都是用来将数组变成一维数组的,并且他们都不会对原数组造成修改,但是 flatten 返回的是一个拷贝,所以对 flatten 的返回值的修改不会影响到原来数组,而 ravel 返回的是一个 View,那么对返回值的修改会影响到原来数组的值

  2. 数据的叠加

    (1) concatenate 可以手动指定 axis 参数具体在哪个方向叠加 (建议使用)

  3. 数组的切割

    (1) split/array_split 是手动的指定 axis 参数,axis=0 代表按行进行切割, axis=1 代表按列进行切割

  4. 矩阵转置

    (1)可以通过 ndarray.T 的形式进行转置

11. View或者浅拷贝

11.1 不拷贝

只是简单的赋值,那么就不会进行拷贝

q9 = np.arange(12)
#这种情况不会进行拷贝
resq9 = q9
#返回True,说明b和a是相同的
print("resq9 is q9: ", resq9 is q9)
# resq9 is q9:  True

11.2 浅拷贝

有些情况,会进行变量的拷贝,但是他们所指向的内存空间都是一样的,那么这种情况叫做浅拷贝,或者叫做View(视图)

q92 = q9.view()
print("q92 is q9: ", q92 is q9)
# q92 is q9:  False
#返回false,说明q92与q9在栈区空间不同,但是所指向的内存空间是一样的

#对q92的值修改 同时也会对a进行修改
q92[0] = 100
print("q92: \n", q92)
# [100   1   2   3   4   5   6   7   8   9  10  11]
print("q9: \n", q9)
# [100   1   2   3   4   5   6   7   8   9  10  11]

11.3 深拷贝

将之前数据完完整整的拷贝一份放到另外一块内存空间中,这样就是两个完全不同的值了

q93 = q9.copy()
print(q93 is q9)
#返回False 说明在不同栈区
#数组d值被修改,数组a值不会被修改 说明内存空间不同
q93[1]=200
print("q9: \n", q9)
print("q93: \n", q93)
# q9: 
#  [100   1   2   3   4   5   6   7   8   9  10  11]
# q93: 
#  [100 200   2   3   4   5   6   7   8   9  10  11]

11.4 总结

在数组操作中分成3种拷贝:

  1. 不拷贝:直接赋值,那么栈区没有拷贝,只是用同一个栈区定义了不同的名称
  2. 浅拷贝:只拷贝栈区,栈区指定的堆区并没有拷贝
  3. 深拷贝:栈区和堆区都拷贝

12. 文件操作

12.1 操作CSV文件

12.1.1 文件保存

np.savetxt(frame,array,fmt="%.18e",delimiter=None)

函数功能:将数组保存到文件中参数说明

  • frame:文件、字符串或产生器,可以是.gz或.bz2的压缩文件
  • array:存入文件的数组
  • fmt:写入文件的格式,例如:%d %.2f %.18e
  • delimter:分割字符串,默认是空格
scores = np.random.randint(0,100,size=(10,2))
print("scores: \n", scores)
# [[73 97]
#  [39 23]
#  [43 21]
#  [22 96]
#  [44  3]
#  [88 63]
#  [59 19]
#  [37 90]
#  [ 8 98]
#  [90 46]]

#保存csv文件
np.savetxt("score.csv",scores,fmt="%d",delimiter=",",header="英语,数学",comments="")
# 英语,数学
# 73,97
# 39,23
# 43,21
# 22,96
# 44,3
# 88,63
# 59,19
# 37,90
# 8,98
# 90,46

12.1.2 读取文件

np.loadtxt(frame,dtype=np.float,delimiter=None,unpack=False)

函数功能:将数组保存到文件中参数说明

  • frame:文件、字符串或产生器,可以是.gz.bz2的压缩文件
  • dtype:数据类型,可选
  • delimiter:分割字符串,默认是任何空格
  • skiprows:跳过前面x
  • usecols:读取指定的列,用元组组合
  • unpack:如果True,读取出来的数组是转置后的
#读取csv文件 跳过第一行的表头
b = np.loadtxt("score.csv",dtype=np.int,delimiter=",",skiprows=1)
print("load txt b: \n", b)
# [[73 97]
#  [39 23]
#  [43 21]
#  [22 96]
#  [44  3]
#  [88 63]
#  [59 19]
#  [37 90]
#  [ 8 98]
#  [90 46]]

12.2 总结

  1. np.savetxt 和 np.loadtxt 一般用来操作 CSV 文件,可以设置 header,但是不能存储3维以上的数组。

  2. np.save 和 np.load 一般用来存储非文本类型的文件,不可以设置 header,但是可以存储3维以上的数组

13. NAN和INF值处理

13.1 介绍

NAN:Not A number,不是一个数字的意思,但是是浮点类型的,所以想要进行数据操作的时候需要注意他的类型

data = np.random.randint(0,10,size=(3,5))
data = data.astype(np.float)
print("data: \n", data)

#将数组中某个位置的值设置为NAN
data[0,1]=np.NAN
data[2,2]=np.NAN
print("data NAN:  \n", data)
# [[ 9. nan  5.  0.  1.]
#  [ 2.  4.  7.  0.  1.]
#  [ 3.  1. nan  2.  0.]]

INF:Infinity,代表的是无穷大的意思,也是属于浮点类型。

np.inf 表示正无穷大,-np.inf 表示负无穷大,一般在出现除数为0的时候为无穷大。比如2/0

13.2 NAN特点

  1. NAN 和 NAN 不相等。比如 np.NAN != np.NAN 这个条件是成立的
  2. NAN 和任何值做运算,结果都是NAN
print("data[0,1] == data[2,2]: \n", data[0,1] == data[2,2])
# Result: False

13.3 处理缺失值

13.3.1 删除缺失值

有时候,我们想要将数组中的NAN删掉,那么我们可以换一种思路,就是只提取不为*NAN*的值

  1. 删除所有NAN的值

因为删除了值后数组将不知道该怎么变化,所以会被变成一维数组

data[~np.isnan(data)]
print("exact not NAN: \n", data[~np.isnan(data)])
#  [9. 5. 0. 1. 2. 4. 7. 0. 1. 3. 1. 2. 0.]
  1. 查找NAN所在行
lines = np.where(np.isnan(data))[0]
print("NAN lines: \n", lines)
# [0 2]

使用delete方法删除指定的行,lines表示删除的行号,axis=0表示删除行

np.delete(data,lines,axis=0)

print("after delete NAN lines data: \n", data )
# [[ 6. nan  5.  7.  5.]
#  [ 6.  2.  2.  3.  8.]
#  [ 1.  1. nan  5.  6.]]

print("after delete NAN lines: \n", np.delete(data,lines, axis=0))
# [[6. 2. 2. 3. 8.]]

13.3.2 用其他值进行替代

data = np.random.randint(0,10,size=(3,5))
data = data.astype(np.float64)
print("data: \n", data)

#将数组中某个位置的值设置为NAN
data[0,1]=np.NAN
data[2,2]=np.NAN
print("data NAN:  \n", data)
#  [[ 4. nan  1.  6.  9.]
#  [ 3.  4.  0.  9.  4.]
#  [ 0.  1. nan  3.  6.]]

data[np.isnan(data)]=0                                                                                
print("将NAN替换为0 : \n", data)
#  [[4. 0. 1. 6. 9.]
#  [3. 4. 0. 9. 4.]
#  [0. 1. 0. 3. 6.]]


13.4 总结

  1. NAN:Not A Number的简写,不是一个数字,但是是属于浮点类型
  2. INF:无穷大,在除数为0的情况下会出现INF
  3. NAN和所有的值进行计算结果都是等于NAN
  4. NAN != NAN
  5. 可以通过np.isnan来判断某个值是不是NAN
  6. 处理值的时候,可以通过删除NAN的形式进行处理,也可以通过值的替换进行处理
  7. np.delete比较特殊,通过axis=0来代表行,而其他大部分函数通过axis=1来代表行

Reference

https://blog.csdn.net/zkx990121/article/details/119136515

posted @ 2023-08-09 22:38  贝壳里的星海  阅读(94)  评论(0编辑  收藏  举报