Python---Numpy万字总结(1)

2024-05-09 13:20
文章标签 python numpy 万字 总结

本文主要是介绍Python---Numpy万字总结(1),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

NumPy的应用(1)

Numpy 是一个开源的 Python 科学计算库,用于快速处理任意维度的数组。Numpy 支持常见的数组和矩阵操作,对于同样的数值计算任务,使用 NumPy 代码简洁,在性能上也远远优于原生 Python,而且数据量越大,NumPy 的优势就越明显。

NumPy 最为核心的数据类型是ndarray,使用ndarray可以处理一维、二维和多维数组,该对象相当于是一个快速而灵活的大数据容器。NumPy 底层代码使用 C 语言编写,解决了 GIL 的限制,ndarray在存取数据的时候,数据与数据的地址都是连续的,这确保了可以进行高效率的批量操作,性能上远远优于 Python 中的list;另一方面ndarray对象提供了更多的方法来处理数据。

准备工作

  1. 启动 JupyterLab

    jupyter lab
    

    提示:在启动 JupyterLab 之前,需要先安装好数据分析相关依赖项,包括三大库函数(Numpy、Pandas、Matplotlib)以及相关依赖项。建议下载 Anaconda,就可以无需单独安装,通过 Anaconda 的 Navigator 来启动。

  2. 导入

    import numpy as np
    import pandas as pd
    import matplotlib.pyplot as plt
    

    说明:若没有安装相关依赖库,可以在单元格中输入%pip install numpy并运行该单元格来安装 NumPy。也可以在单元格中输入%pip install numpy pandas matplotlib把 Python 数据分析三个核心的三方库都安装上。注意上面的代码,不仅导入了 NumPy,还将 pandas 和 matplotlib 库一并导入了。

创建数组对象

创建ndarray对象有很多种方法,下面则是一些常用方法。

方法一:使用array函数,通过list创建数组对象

代码:

array1 = np.array([1, 2, 3])
array1

输出:

array([1, 2, 3])

代码:

array2 = np.array([[1, 2], [3,4]])
array2

输出:

array([[1, 2],[3,4]])

方法二:使用arange函数,指定取值范围和跨度创建数组对象

代码:

array3 = np.arange(0, 10, 2)
array3

输出:

array([ 0,  2,  4,  6,  8])

方法三:使用linspace函数,用指定范围和元素个数创建数组对象,生成等差数列

代码:

array4 = np.linspace(-1, 1, 11)
array4

输出:

array([-1. , -0.8, -0.6, -0.4, -0.2,  0. ,  0.2,  0.4,  0.6,  0.8,  1. ])

方法四:使用logspace函数,生成等比数列

代码:

array5 = np.logspace(1, 10, num=10, base=2)
array5

注意:等比数列的起始值是 2 1 2^1 21,等比数列的终止值是 2 10 2^{10} 210num是元素的个数,base就是底数。

输出:

array([   2.,    4.,    8.,   16.,   32.,   64.,  128.,  256.,  512., 1024.])

方法五:通过fromstring函数从字符串提取数据创建数组对象

代码:

array6 = np.fromstring('1, 2, 3, 4, 5', sep=',', dtype='i8')
array6 

输出:

array([1, 2, 3, 4, 5])

方法六:通过fromiter函数从生成器(迭代器)中获取数据创建数组对象

代码:

def fib(how_many):a, b = 0, 1for _ in range(how_many):a, b = b, a + byield agen = fib(20)
array7 = np.fromiter(gen, dtype='i8')
array7

输出:

array([   1,    1,    2,    3,    5,    8,   13,   21,   34,   55,   89,144,  233,  377,  610,  987, 1597, 2584, 4181, 6765])

方法七:使用numpy.random模块的函数生成随机数创建数组对象

产生10个 [ 0 , 1 ) [0, 1) [0,1)范围的随机小数,代码:

array8 = np.random.rand(10)
array8

输出:

array([0.45556132, 0.67871326, 0.4552213 , 0.96671509, 0.44086463,0.72650875, 0.79877188, 0.12153022, 0.24762739, 0.6669852 ])

产生10个 [ 1 , 100 ) [1, 100) [1,100)范围的随机整数,代码:

array9 = np.random.randint(1, 100, 10)
array9

输出:

array([29, 97, 87, 47, 39, 19, 71, 32, 79, 34])

产生20个 μ = 50 \small{\mu=50} μ=50 σ = 10 \small{\sigma=10} σ=10的正态分布随机数,代码:

array10 = np.random.normal(50, 10, 20)
array10

输出:

array([55.04155586, 46.43510797, 20.28371158, 62.67884053, 61.23185964,38.22682148, 53.17126151, 43.54741592, 36.11268017, 40.94086676,63.27911699, 46.92688903, 37.1593374 , 67.06525656, 67.47269463,23.37925889, 31.45312239, 48.34532466, 55.09180924, 47.95702787])

产生[0, 1)范围的随机小数构成的3行4列的二维数组,代码:

array11 = np.random.rand(3, 4)
array11

输出:

array([[0.54017809, 0.46797771, 0.78291445, 0.79501326],[0.93973783, 0.21434806, 0.03592874, 0.88838892],[0.84130479, 0.3566601 , 0.99935473, 0.26353598]])

产生[1, 100)范围的随机整数构成的三维数组,代码:

array12 = np.random.randint(1, 100, (3, 4, 5))
array12

输出:

array([[[94, 26, 49, 24, 43],[27, 27, 33, 98, 33],[13, 73,  6,  1, 77],[54, 32, 51, 86, 59]],[[62, 75, 62, 29, 87],[90, 26,  6, 79, 41],[31, 15, 32, 56, 64],[37, 84, 61, 71, 71]],[[45, 24, 78, 77, 41],[75, 37,  4, 74, 93],[ 1, 36, 36, 60, 43],[23, 84, 44, 89, 79]]])

方法八:创建全0、全1或指定元素的数组

使用zeros函数,代码:

array13 = np.zeros((3, 4))
array13

输出:

array([[0., 0., 0., 0.],[0., 0., 0., 0.],[0., 0., 0., 0.]])

使用ones函数,代码:

array14 = np.ones((3, 4))
array14

输出:

array([[1., 1., 1., 1.],[1., 1., 1., 1.],[1., 1., 1., 1.]])

使用full函数创建一个三行四列全为10的数二维数组。代码:

array15 = np.full((3, 4), 10)
array15

输出:

array([[10, 10, 10, 10],[10, 10, 10, 10],[10, 10, 10, 10]])

方法九:使用eye函数创建单位矩阵

代码:

np.eye(4)

输出:

array([[1., 0., 0., 0.],[0., 1., 0., 0.],[0., 0., 1., 0.],[0., 0., 0., 1.]])

方法十:读取图片获得对应的三维数组

代码:

array16 = plt.imread('abc/test.jpg')
array16

输出:

array([[[ 36,  33,  28],[ 36,  33,  28],[ 36,  33,  28],...,[ 32,  31,  29],[ 32,  31,  27],[ 31,  32,  26]],[[ 37,  34,  29],[ 38,  35,  30],[ 38,  35,  30],...,[ 31,  30,  28],[ 31,  30,  26],[ 30,  31,  25]],[[ 38,  35,  30],[ 38,  35,  30],[ 38,  35,  30],...,[ 30,  29,  27],[ 30,  29,  25],[ 29,  30,  25]],...,[[239, 178, 123],[237, 176, 121],[235, 174, 119],...,[ 78,  68,  56],[ 75,  67,  54],[ 73,  65,  52]],[[238, 177, 120],[236, 175, 118],[234, 173, 116],...,[ 82,  70,  58],[ 78,  68,  56],[ 75,  66,  51]],[[238, 176, 119],[236, 175, 118],[234, 173, 116],...,[ 84,  70,  61],[ 81,  69,  57],[ 79,  67,  53]]], dtype=uint8)

说明:上面的代码读取了当前路径下abc目录中名为test.jpg 的图片文件,计算机系统中的图片通常由若干行若干列的像素点构成,而每个像素点又是由红绿蓝三原色构成的,刚好可以用三维数组来表示。读取图片用到了matplotlib库的imread函数。

数组对象的属性

size属性:获取数组元素个数。

代码:

array17 = np.arange(1, 100, 2)
array18 = np.random.rand(3, 4)
print(array16.size)
print(array17.size)
print(array18.size)

输出:

1125000
50
12

shape属性:获取数组的形状。

代码:

print(array16.shape)
print(array17.shape)
print(array18.shape)

输出:

(750, 500, 3)
(50,)
(3, 4)

dtype属性:获取数组元素的数据类型。

代码:

print(array16.dtype)
print(array17.dtype)
print(array18.dtype)

输出:

uint8
int64
float64

ndarray对象元素的数据类型可以参考如下所示的表格。
在这里插入图片描述

ndim属性:获取数组的维度。

代码:

print(array16.ndim)
print(array17.ndim)
print(array18.ndim)

输出:

3
1
2

itemsize属性:获取数组单个元素占用内存空间的字节数。

代码:

print(array16.itemsize)
print(array17.itemsize)
print(array18.itemsize)

输出:

1
8
8

nbytes属性:获取数组所有元素占用内存空间的字节数。

代码:

print(array16.nbytes)
print(array17.nbytes)
print(array18.nbytes)

输出:

1125000
400
96

数组的索引运算

和 Python 中的列表类似,NumPy 的ndarray对象可以进行索引和切片操作,通过索引可以获取或修改数组中的元素,通过切片操作可以取出数组的一部分,我们把切片操作也称为切片索引。

普通索引

类似于 Python 中list类型的索引运算。

代码:

array19 = np.arange(1, 10)
print(array19[0], array19[array19.size - 1])
print(array19[-array20.size], array19[-1])

输出:

1 9
1 9

代码:

array20 = np.array([[1, 2, 3], [4, 5, 6], [7, 8, 9]])
array20[2]

输出:

array([7, 8, 9])

代码:

print(array20[0][0])
print(array20[-1][-1])

输出:

1
9

代码:

print(array20[1][1])
print(array20[1, 1])

输出:

5
5

代码:

array20[1][1] = 10
array20

输出:

array([[ 1,  2,  3],[ 4, 10,  6],[ 7,  8,  9]])

代码:

array20[1] = [10, 11, 12]
array20

输出:

array([[ 1,  2,  3],[10, 11, 12],[ 7,  8,  9]])
切片索引

切片索引是形如[开始索引:结束索引:跨度]的语法,通过指定开始索引(默认值无穷小)、结束索引(默认值无穷大)和跨度(默认值1),从数组中取出指定部分的元素并构成新的数组。因为开始索引、结束索引和步长都有默认值,所以它们都可以省略,如果不指定步长,第二个冒号也可以省略。一维数组的切片运算跟 Python 中的list类型的切片非常类似。

代码:

array20[:2, 1:]

输出:

array([[ 2,  3],[11, 12]])

代码:

array20[2, :]

输出:

array([7, 8, 9])

代码:

array20[2:, :]

输出:

array([[7, 8, 9]])

代码:

array20[:, :2]

输出:

array([[ 1,  2],[10, 11],[ 7,  8]])

代码:

array20[::2, ::2]

输出:

array([[1, 3],[7, 9]])

代码:

array20[::-2, ::-2]

输出:

array([[9, 7],[3, 1]])

关于数组的索引和切片运算,可以通过下面的两张图来加深理解,图片来自《利用Python进行数据分析》一书

图1:二维数组的普通索引

在这里插入图片描述

图2:二维数组的切片索引

在这里插入图片描述

花式索引

花式索引是用保存整数的数组充当一个数组的索引,这里所说的数组可以是 NumPy 的ndarray,也可以是 Python 中listtuple等可迭代类型,可以使用正向或负向索引。

代码:

array19[[0, 1, 1, -1, 4, -1]]

输出:

array([1, 2, 2, 9, 5, 9])

代码:

array20[[0, 2]]

输出:

array([[1, 2, 3],[7, 8, 9]])

代码:

array20[[0, 2], [1, 2]]

输出:

array([2, 9])

代码:

array20[[0, 2], 1]

输出:

array([2, 8])
布尔索引

布尔索引就是通过保存布尔值的数组充当一个数组的索引,布尔值为True的元素保留,布尔值为False的元素不会被选中。布尔值的数组可以手动构造,也可以通过关系运算来产生。

代码:

array19[[True, True, False, False, True, False, False, True, True]]

输出:

array([1, 2, 5, 8, 9])

代码:

array19 > 5

输出:

array([False, False, False, False, False,  True,  True,  True,  True])

代码:

~(array19 > 5)

输出:

array([ True,  True,  True,  True,  True, False, False, False, False])

代码:

array19[array20 > 5]

输出:

array([6, 7, 8, 9])

代码:

array19 % 2 == 0

输出:

array([False,  True, False,  True, False,  True, False,  True, False])

代码:

array19[array20 % 2 == 0]

输出:

array([2, 4, 6, 8])

代码:

(array19 > 5) & (array19 % 2 == 0)

输出:

array([False, False, False, False, False,  True, False,  True, False])

说明&运算符可以作用于两个布尔数组,如果两个数组对应元素都是True,那么运算的结果就是True,否则就是False,该运算符的运算规则类似于 Python 中的 and 运算符,只不过作用的对象是两个布尔数组。

代码:

array19[(array19 > 5) & (array19 % 2 == 0)]

输出:

array([6, 8])

代码:

array19[(array19 > 5) | (array19 % 2 == 0)]

输出:

array([2, 4, 6, 7, 8, 9])

说明|运算符可以作用于两个布尔数组,如果两个数组对应元素都是False,那么运算的结果就是False,否则就是True,该运算符的运算规则类似于 Python 中的 or 运算符,只不过作用的对象是两个布尔数组。

代码:

array20[array21 % 2 != 0]

输出:

array([1, 3, 5, 7, 9])

在Python中进行索引操作时,要注意索引从0开始,支持负索引和切片,适用于可迭代对象。列表等可变类型可进行赋值,而元组和字符串等不可变类型则只能读取。多维索引用于嵌套结构,布尔索引和列表推导可用于复杂操作。同时,注意索引越界会引发错误,且硬编码索引应避免,以提高代码的可读性和维护性。

案例:通过数组切片处理图像

学习基础知识总是比较枯燥且没有成就感的,所以我们还是来个案例为大家演示下上面学习的数组索引和切片操作到底有什么用。前面我们说到过,可以用三维数组来表示图像,那么通过图像对应的三维数组进行操作,就可以实现对图像的处理,如下所示。

读入图片创建三维数组对象。

image1 = plt.imread('man.png')
plt.imshow(image1)

在这里插入图片描述

对数组的0轴进行反向切片,实现图像的垂直翻转。

plt.imshow(image1[::-1])

在这里插入图片描述

对数组的1轴进行反向切片,实现图像的水平翻转。

plt.imshow(guido_image[:,::-1])

在这里插入图片描述

通过切片操作实现抠图,将其中一个人给抠出来。

plt.imshow(image1[30:350, 90:300])

在这里插入图片描述

通过切片操作实现降采样。

plt.imshow(image1[::10, ::10])

在这里插入图片描述

这篇关于Python---Numpy万字总结(1)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/973530

相关文章

Android图片轮播的实现总结

前言: 在很多app中,我们都可以看到几张图片每隔一段时间就切换一下,这就是我们所称的图片轮播的功能,其主要实现就是用到了ViewPager, 下面我们来着重讲解一下其具体实现 效果图: 步骤一:在XML中添加ViewPager控件 比如: <?xml version="1.0" encoding="utf-8"?><RelativeLayout xmlns:a

自定义ViewGroup的总结(侧滑特效)

前言: 和自定义View控件一样,我们有时也需要自定义我们想要的ViewGroup,那么此时,我们就需要让其继承ViewGroup,然后重写 里边的onMeasure()和onLayout()方法,下面以侧滑特效为例,来讲解一下自定义ViewGroup所需的流程,关于侧滑特效, 其整体效果图如下: 对于自定义ViewGroup,主要有以下几步: 步骤一:编写ViewGroup

自定义View的总结(自定义滑动开关)

前言: 由于有些控件,在android中样式比较挫,并不能满足我们的需求,此时,我们可以将其进行一个自定义,下面一以一个自定义编写的ToggleButton为例, 来简要说明下,自定义所涉及到的一些步骤;以下是自定义控件ToggleButton的效果图: 其是由两张图片组成的: <1>   <2>  下面我们通过这个示例,来说明下,如何编写一个自定义view控件!!

关于隐藏Android标题栏总结

1>区分状态栏/标题栏/导航栏 状态栏(Status Bar) 标题栏(Title Bar) 导航栏(Navigation Bar) 2>区分Title Bar/Action Bar/Tool Bar Title Bar就是我们所俗称的标题栏,在Android 3.0 (API level 11)的时候,引入的Action Bar,其就是用来取代Title Bar的,

Android的Paint和Canvas的使用总结

前言: 在自定义控件时,我们有时可能会用到Paint和Canvas这两个类, Paint相当于我们在画画时的画笔,Canvs相当于我们在画画时的画布, 下面来简单讲一下这两个类常见的一些用法 Paint的使用总结: setAlpha(int a): 设置画笔的透明度,这样画笔所画的位置就会呈一定的透明度 setAntiAlias(boolean aa): 设置 tr

Android的FragmentTabHost使用总结(顶部或底部菜单栏)

前言: 我们经常看到一些app的自带一些标签,并且可以来回进行切换, 本章我们就通过FragmentTabHost来学习一下其如何实现,效果图如下: 步骤一: 编写布局文件 <android.support.v4.app.FragmentTabHost android:layout_width="match_parent" android:la

Android的RecyclerView使用总结

前言: RecyclerView 小组件比 ListView 更高级且更具灵活性。 此小组件是一个用于显示庞大数据集的容器,可通过保持有限数量的视图进行非常有效的滚动操作。 如果您有数据集合,其中的元素将因用户操作或网络事件而发生改变,请使用RecyclerView 小组件。 RecyclerView  类别将通过提供下列功能简化庞大数据集的显示与处理: 用于项目定位的布局管理器 用于通用项目

java基础—java中使用final关键字的总结

有时候我,们希望某些东西是亘古不变的,可以使用final关键字完成这个重任! final学习总结: 1:final + 属性 如果属性是基本数据类型(byte 字节型short 短整型int 普通整型char 字符型float 浮点型long 长整型double 双精度),则变为常量,其值不能被更改;如果属性是引用类型,则引用地址不能被更改。(final 修饰一个对象,那么这个对象的引用地址

总结这段时间的学习和生活

越来越发现自己活的很狭隘,思想也比较狭隘! 现在的生活是8点每天早上起床,在家吃个早饭,然后骑车去公司。这几天发现走路去公司也就十几分钟,那就走路吧,走路的时候自己也听一听音乐或者一些线上的音频文件,发现早晨走路还是听舒服的,这样也可以简单的锻炼一下身体,从惠州到长沙,很久没有坚持锻炼了。 为什么没有坚持锻炼呢,我给自己找了一些理由,一个是住的地方不方便,一个刚加入新的公司要努力学习,尽

有效管理自己知识,多总结和分享——2018七月份的尾巴

知道却做不到,等于不知道! --笔记侠 知道却讲不出来,也等于不知道! --dufy 知道 ≠ 能做到 | 知道 ≠ 能讲出 ? 昨天听了一个音频,里面说:“知道很多道理 却依然过不好这一生”! 知道了这么多道理,能真正做到有多少呢?知道了这么多道理,真正用自己的理解或者自己的话能表达的道理又有多少呢? 我在反思自己,希望与你共勉! 分享:好的学习方法 — 费曼技巧 知道只是一个