利用python进行数据分析第四章(NumPy基础:数组与向量化计算)

  • NumPy(Numerical Python的简称)是Python数值计算最重要的基础包。大多数提供科学计算的包都是用NumPy的数组作为构建基础。
  • NumPy的部分功能如下:
    • ndarray,一个具有矢量算术运算和复杂广播能力的快速且节省空间的多维数组。
    • 用于对整组数据进行快速运算的标准数学函数(无需编写循环)。
    • 用于读写磁盘数据的工具以及用于操作内存映射文件的工具。
    • 线性代数、随机数生成以及傅里叶变换功能。
    • 用于集成由C、C++、Fortran等语言编写的代码的A C API。
  • 由于NumPy提供了一个简单易用的C API,因此很容易将数据传递给由低级语言编写的外部库,外部库也能以NumPy数组的形式将数据返回给Python。这个功能使Python成为一种包装C/C++/Fortran历史代码库的选择,并使被包装库拥有一个动态的、易用的接口。
  • NumPy本身并没有提供多么高级的数据分析功能,理解NumPy数组以及面向数组的计算将有助于你更加高效地使用诸如pandas之类的工具。因为NumPy是一个很大的题目,我会在附录A中介绍更多NumPy高级功能,比如广播。
  • 对于大部分数据分析应用而言,我最关注的功能主要集中在:
    • 用于数据整理和清理、子集构造和过滤、转换等快速的矢量化数组运算。
    • 常用的数组算法,如排序、唯一化、集合运算等。
    • 高效的描述统计和数据聚合/摘要运算。
    • 用于异构数据集的合并/连接运算的数据对齐和关系型数据运算。
    • 将条件逻辑表述为数组表达式(而不是带有if-elif-else分支的循环)。
    • 数据的分组运算(聚合、转换、函数应用等)。。
  • 虽然NumPy提供了通用的数值数据处理的计算基础,但大多数读者可能还是想将pandas作为统计和分析工作的基础,尤其是处理表格数据时。pandas还提供了一些NumPy所没有的领域特定的功能,如时间序列处理等。

笔记:Python的面向数组计算可以追溯到1995年,Jim Hugunin创建了Numeric库。接下来的10年,许多科学编程社区纷纷开始使用Python的数组编程,但是进入21世纪,库的生态系统变得碎片化了。2005年,Travis Oliphant从Numeric和Numarray项目整了出了NumPy项目,进而所有社区都集合到了这个框架下。

  • NumPy之于数值计算特别重要的原因之一,是因为它可以高效处理大数组的数据。这是因为:
    • NumPy是在一个连续的内存块中存储数据,独立于其他Python内置对象。NumPy的C语言编写的算法库可以操作内存,而不必进行类型检查或其它前期工作。比起Python的内置序列,NumPy数组使用的内存更少。
    • NumPy可以在整个数组上执行复杂的计算,而不需要Python的for循环。
  • 要搞明白具体的性能差距,考察一个包含一百万整数的数组,和一个等价的Python列表:
    在这里插入图片描述
  • 现在我们同时对每个序列乘以2:
    在这里插入图片描述
  • NumPy的方法比Python方法要快10到100倍,并且使用的内存也更少。

4.1 NumPy ndarray:多维数组对象

  • NumPy最重要的一个特点就是其N维数组对象(即ndarray),该对象是一个快速而灵活的大数据集容器。你可以利用这种数组对整块数据执行一些数学运算,其语法跟标量元素之间的运算一样。
  • 要明白Python是如何利用与标量值类似的语法进行批次计算,我先引入NumPy,然后生成一个包含随机数据的小数组:
    在这里插入图片描述
  • 然后给data加上一个数学操作:
    在这里插入图片描述
  • 在第一个数学操作中,所有的元素都同时乘了10。在第二个数学操作中,数组中的对应元素进行了相加。

在本章及全书中,我会使用标准的NumPy惯用法import numpy as np。你当然也可以在代码中使用from numpy import *,但不建议这么做。numpy的命名空间很大,包含许多函数,其中一些的名字与Python的内置函数重名(比如min和max)。

  • ndarray是一个通用的同构数据多维容器,也就是说,其中的所有元素必须是相同类型的。每个数组都有一个shape(一个表示各维度大小的元组)和一个dtype(一个用于说明数组数据类型的对象):
    在这里插入图片描述
  • 本章将会介绍NumPy数组的基本用法,这对于本书后面各章的理解基本够用。虽然大多数数据分析工作不需要深入理解NumPy,但是精通面向数组的编程和思维方式是成为Python科学计算牛人的一大关键步骤。

当你在本书中看到“数组”、“NumPy数组”、”ndarray”时,基本上都指的是同一样东西,即ndarray对象。

4.1.1 生成ndarray

  • 创建数组最简单的办法就是使用array函数。它接受一切序列型的对象(包括其他数组),然后产生一个新的含有传入数据的NumPy数组。以一个列表的转换为例:
    在这里插入图片描述
  • 嵌套序列(比如由一组等长列表组成的列表)将会被转换为一个多维数组:
    在这里插入图片描述
  • 因为data2是列表的列表,NumPy数组arr2的两个维度的shape是从data2引入的。可以用属性ndim和shape验证:
    在这里插入图片描述
  • 除非特别说明(稍后将会详细介绍),np.array会尝试为新建的这个数组推断出一个较为合适的数据类型。数据类型保存在一个特殊的dtype对象中。比如说,在上面的两个例子中,我们有:
    在这里插入图片描述
  • 除np.array之外,还有一些函数也可以新建数组。比如,zeros和ones分别可以创建指定长度或形状的全0或全1数组。empty可以创建一个没有任何具体值的数组。要用这些方法创建多维数组,只需传入一个表示形状的元组即可:
    在这里插入图片描述

认为np.empty会返回全0数组的想法是不安全的。很多情况下(如前所示),它返回的都是一些未初始化的垃圾值。

  • arange是Python内置函数range的数组版:
    在这里插入图片描述
  • 表4-1列出了一些数组创建函数。由于NumPy关注的是数值计算,因此,如果没有特别指定,数据类型基本都是float64(浮点数)
    在这里插入图片描述

4.1.2 ndarray 的数组类型

  • dtype(数据类型)是一个特殊的对象,它含有ndarray将一块内存解释为特定数据类型所需的信息:
    在这里插入图片描述
  • dtype是NumPy灵活交互其它系统的源泉之一。多数情况下,它们直接映射到相应的机器表示,这使得“读写磁盘上的二进制数据流”以及“集成低级语言代码(如C、Fortran)”等工作变得更加简单。数值型dtype的命名方式相同:一个类型名(如float或int),后面跟一个用于表示各元素位长的数字。标准的双精度浮点值(即Python中的float对象)需要占用8字节(即64位)。因此,该类型在NumPy中就记作float64。表4-2列出了NumPy所支持的全部数据类型。

记不住这些NumPy的dtype也没关系,新手更是如此。通常只需要知道你所处理的数据的大致类型是浮点数、复数、整数、布尔值、字符串,还是普通的Python对象即可。当你需要控制数据在内存和磁盘中的存储方式时(尤其是对大数据集),那就得了解如何控制存储类型。
在这里插入图片描述
在这里插入图片描述

  • 你可以使用astype 方法显式地转换数组的数据类型:
    在这里插入图片描述
  • 在本例中,整数被转换成了浮点数。如果将浮点数转换成整数,则小数部分将会被截取删除:
    在这里插入图片描述
  • 如果你有一个数组,里面的元素都是表达数字意义的字符串,也可以通过astype将字符串转换为数字:
    在这里插入图片描述

使用numpy.string_类型时,一定要小心,因为NumPy会修改它的大小或删除输入且不发生警告。pandas提供了更多非数值数据的便利的处理方法。

  • 如果转换过程因为某种原因而失败了(比如某个不能被转换为float64的字符串),就会引发一个ValueError。这里,我比较懒,写的是float而不是np.float64;NumPy很聪明,它会将Python类型映射到等价的dtype上。
  • 你也可以使用另一个数组的dtype属性:
    在这里插入图片描述
  • 也可以使用类型代码来传入数据类型:
    在这里插入图片描述

调用astype总会创建一个新的数组(一个数据的备份),即使新的dtype与旧的dtype相同。

4.1.3 NumPy数组算术

  • 数组很重要,因为它使你不用编写循环即可对数据执行批量运算。NumPy用户称其为矢量化(vectorization)。大小相等的数组之间的任何算术运算都会将运算应用到元素级:
    在这里插入图片描述
  • 带有标量计算的算术操作,会把计算参数传递给数组的每一个元素:
    在这里插入图片描述
  • 同尺寸数组之间的比较,会产生一个布尔值数组:
    在这里插入图片描述
  • 不同大小的数组之间的运算叫做广播(broadcasting),将在附录A中对其进行详细讨论。本书的内容不需要对广播机制有多深的理解

4.1.4 基础索引与切片

  • NumPy数组的索引是一个内容丰富的主题,因为选取数据子集或单个元素的方式有很多。一维数组很简单。从表面上看,它们跟Python列表的功能差不多:
    在这里插入图片描述

  • 如上所示,当你将一个标量值赋值给一个切片时(如arr[5:8]=12),该值会自动传播(也就说后面将会讲到的“广播”)到整个选区。跟列表最重要的区别在于,数组切片是原始数组的视图。这意味着数据不会被复制,视图上的任何修改都会直接反映到源数组上。

  • 例子如下:
    在这里插入图片描述

  • 当我改变arr_slice,变化也会体体现在原数组上:
    在这里插入图片描述

  • 不写切片值的[:]将会引用数组的所有值:
    在这里插入图片描述

  • 如果你刚开始接触NumPy,可能会对此感到惊讶(尤其是当你曾经用过其他热衷于复制数组数据的编程语言)。由于NumPy的设计目的是处理大数据,所以你可以想象一下,假如NumPy坚持要将数据复制来复制去的话会产生何等的性能和内存问题。

如果你想要得到的是ndarray切片的一份副本而非视图,就需要明确地进行复制操作,例如arr[5:8].copy()

  • 对于高维度数组,能做的事情更多。在一个二维数组中,每个索引值对应的元素不再是一个值,而是一个一维数组:
    在这里插入图片描述
  • 因此,可以对各个元素进行递归访问,但这样需要做的事情有点多。你可以传入一个以逗号隔开的索引列表来选取单个元素。也就是说,下面两种方式是等价的:
    在这里插入图片描述
  • 图4-1说明了二维数组的索引方式。轴0作为行,轴1作为列。
    在这里插入图片描述
  • 在多维数组中,如果省略了后面的索引,则返回对象会是一个维度低一点的ndarray(它含有高一级维度上的所有数据)。因此,在2×2×3数组arr3d中:
    在这里插入图片描述
  • arr3d[0]是一个2×3数组:
    在这里插入图片描述
  • 标量和数组都可以传递给arr3d[0]:
    在这里插入图片描述
  • 类似地,arr3d[1,0]可以访问索引以(1,0)开头的那些值(以一维数组的形式返回):
    在这里插入图片描述
  • 上面的表达式可以分解为下面两步:
    在这里插入图片描述
  • 需要注意的是,以上的数组子集选择中,返回的数组都是视图:
数组的切片索引
  • ndarray的切片语法跟Python列表这样的一维对象差不多:
    在这里插入图片描述

  • 再回想下前面的二维数组,arr2d,对数组进行切片略有不同:
    在这里插入图片描述

  • 可以看出,它是沿着第0轴(即第一个轴)切片的。也就是说,切片是沿着一个轴向选取元素的。表达式arr2d[:2]可以被认为是“选取arr2d的前两行”

  • 你可以一次传入多个切片,就像传入多个索引那样:
    在这里插入图片描述

  • 类似地,我也可以选择第三列,但是只选择前两行:
    在这里插入图片描述

  • 图4-2对此进行了说明。注意,“只有冒号”表示选取整个轴,因此你可以像下面这样只对高维轴进行切片:
    在这里插入图片描述
    在这里插入图片描述

4.1.5 布尔索引

  • 来看这样一个例子,假设我们有一个用于存储数据的数组以及一个存储姓名的数组(含有重复项)。在这里,我将使用numpy.random中的randn函数生成一些正态分布的随机数据:
    在这里插入图片描述
  • 假设每个名字都对应data数组中的一行,而我们想要选出对应于名字”Bob”的所有行。跟算术运算一样,数组的比较运算(如==)也是矢量化的。因此,对names和字符串”Bob”的比较运算将会产生一个布尔型数组:
    在这里插入图片描述
  • 在索引数组时可以传入布尔值数组:
    在这里插入图片描述
  • 布尔型数组的长度必须跟被索引的轴长度一致。此外,还可以将布尔型数组跟切片、整数(或整数序列,稍后将对此进行详细讲解)混合使用

如果布尔型数组的长度不对,布尔型选择就会出错,因此一定要小心。

  • 下面的例子,我选取了names == 'Bob’的行,并索引了列:
    在这里插入图片描述
  • 要选择除”Bob”以外的其他值,既可以使用不等于符号(!=),也可以通过~对条件进行否定:
    在这里插入图片描述
  • ~操作符可以在你想要对一个通用条件进行取反时使用:
    在这里插入图片描述
  • 选取这三个名字中的两个需要组合应用多个布尔条件,使用&(和)、|(或)之类的布尔算术运算符即可:
    在这里插入图片描述
  • 通过布尔型索引选取数组中的数据,将总是创建数据的副本,即使返回一模一样的数组也是如此。

注意:Python关键字and和or在布尔型数组中无效。要使用&与|。

  • 通过布尔型数组设置值是一种经常用到的手段。为了将data中的所有负值都设置为0,我们只需:
    在这里插入图片描述
  • 利用一维布尔值数组对每一行设置数值也是非常简单的:
    在这里插入图片描述

4.1.6 神奇索引

  • 神奇索引是一个NumPy术语,它指的是利用整数数组进行索引。假设我们有一个8×4数组:
    在这里插入图片描述
  • 为了以特定顺序选取行子集,只需传入一个用于指定顺序的整数列表或ndarray即可:
    在这里插入图片描述
  • 代码运行出你想要的结果!如果使用负的索引,将从尾部进行选择:
    在这里插入图片描述
  • 一次传入多个索引数组会有一点特别。它返回的是一个一维数组,其中的元素对应各个索引元组:
    在这里插入图片描述
  • 在附录A中可以看到reshape方法的更多细节。
    在这里插入图片描述
  • 最终选出的是元素(1,0)、(5,3)、(7,1)和(2,2)。无论数组是多少维的,花式索引总是一维的。
  • 这个花式索引的行为可能会跟某些用户的预期不一样(包括我在内),选取矩阵的行列子集应该是矩形区域的形式才对。下面是得到该结果的一个办法:
    在这里插入图片描述
  • 记住,花式索引跟切片不一样,它总是将数据复制到新数组中。

4.1.7 数组转置和转轴

  • 转置是重塑的一种特殊形式,它返回的是源数据的视图(不会进行任何复制操作)。数组不仅有transpose方法,还有一个特殊的T属性:
    在这里插入图片描述

  • 在进行矩阵计算时,经常需要用到该操作,比如利用np.dot计算矩阵内积:
    在这里插入图片描述

  • 对于高维数组,transpose需要得到一个由轴编号组成的元组才能对这些轴进行转置(比较费脑子):
    在这里插入图片描述

  • 这里,第一个轴被换成了第二个,第二个轴被换成了第一个,最后一个轴不变。

  • 简单的转置可以使用.T,它其实就是进行轴对换而已。ndarray还有一个swapaxes方法,它需要接受一对轴编号:
    在这里插入图片描述

  • swapaxes也是返回源数据的视图(不会进行任何复制操作)。

4.2 通用函数:快速的逐元素数组函数

  • 通用函数(即ufunc)是一种对ndarray中的数据执行元素级运算的函数。你可以将其看做简单函数(接受一个或多个标量值,并产生一个或多个标量值)的矢量化包装器。

  • 有很多ufunc是简单的逐元素转换,比如sqrt或exp函数:
    在这里插入图片描述

  • 这些都是一元(unary)ufunc。另外一些(如add或maximum)接受2个数组(因此也叫二元(binary)ufunc),并返回一个结果数组:
    在这里插入图片描述

  • 这里,numpy.maximum计算了x和y中元素级别最大的元素。

  • 虽然并不常见,但有些ufunc的确可以返回多个数组。modf就是一个例子,它是Python内置函数divmod的矢量化版本,它会返回浮点数数组的小数和整数部分:

在这里插入图片描述

  • 通用函数接收一个可选参数out,允许对数组进行位置操作:
    在这里插入图片描述
  • 表4-3和表4-4列举的是可用的通用函数
    在这里插入图片描述
    在这里插入图片描述
    在这里插入图片描述

4.3 使用数组进行面向数组编程

  • NumPy数组使你可以将许多种数据处理任务表述为简洁的数组表达式(否则需要编写循环)。用数组表达式代替循环的做法,通常被称为矢量化。一般来说,矢量化数组运算要比等价的纯Python方式快上一两个数量级(甚至更多),尤其是各种数值计算。在后面内容中(见附录A)我将介绍广播,这是一种针对矢量化计算的强大手段。

  • 作为简单的例子,假设我们想要在一组值(网格型)上计算函数sqrt(x2+y2)。np.meshgrid函数接受两个一维数组,并产生两个二维矩阵(对应于两个数组中所有的(x,y)对):
    在这里插入图片描述

  • 现在,你可以用和两个坐标值同样的表达式来使用函数:
    在这里插入图片描述

  • 作为第九章的前瞻,我使用matplotlib来生成这个二维数组的可视化:

In [160]: import matplotlib.pyplot as plt
In [161]: plt.imshow(z, cmap=plt.cm.gray); plt.colorbar()
Out[161]: <matplotlib.colorbar.Colorbar at 0x7f715e3fa630>
In [162]: plt.title("Image plot of $\sqrt{x^2 + y^2}$ for a grid of values")
Out[162]: <matplotlib.text.Text at 0x7f715d2de748>
  • 见图4-3。这张图是用matplotlib的imshow函数创建的。
    在这里插入图片描述

4.3.1 将条件逻辑作为数组操作

  • numpy.where函数是三元表达式x if condition else y的矢量化版本。假设我们有一个布尔数组和两个值数组:
    在这里插入图片描述
  • 假设cond中的元素为True时,我们取xarr中的对应元素值,否则取yarr中的元素。我们可以通过列表推导式来完成,向下面代码这样:
    在这里插入图片描述
  • 这有几个问题。第一,它对大数组的处理速度不是很快(因为所有工作都是由纯Python完成的)。第二,无法用于多维数组。若使用np.where,则可以将该功能写得非常简洁:
    在这里插入图片描述
  • np.where的第二个和第三个参数不必是数组,它们都可以是标量值。在数据分析工作中,where通常用于根据另一个数组而产生一个新的数组。假设有一个由随机数据组成的矩阵,你希望将所有正值替换为2,将所有负值替换为-2。若利用np.where,则会非常简单:
    在这里插入图片描述
  • 你可以使用np.where将标量和数组联合,例如,我可以像下面的代码那样将arr中的所有正值替换为常数2:
    在这里插入图片描述
  • 传递给np.where的数组既可以是同等大小的数组,也可以是标量。

4.3.2 数学和统计方法

  • 可以通过数组上的一组数学函数对整个数组或某个轴向的数据进行统计计算。sum、mean以及标准差std等聚合计算(aggregation,通常叫做约简(reduction))既可以当做数组的实例方法调用,也可以当做顶级NumPy函数使用。

  • 这里,我生成了一些正态分布随机数据,然后做了聚类统计:
    在这里插入图片描述

  • mean和sum这类的函数可以接受一个axis选项参数,用于计算该轴向上的统计值,最终结果是一个下降一维的数组:
    在这里插入图片描述

  • 这里,arr.mean(1)是“计算行的平均值”,arr.sum(0)是“计算每列的和”。

  • 其他如cumsum和cumprod之类的方法则不聚合,而是产生一个由中间结果组成的数组:
    在这里插入图片描述

  • 在多维数组中,累加函数(如cumsum)返回的是同样大小的数组,但是会根据每个低维的切片沿着标记轴计算部分聚类:
    在这里插入图片描述

  • 表4-5列出了全部的基本数组统计方法。后续章节中有很多例子都会用到这些方法。
    在这里插入图片描述
    在这里插入图片描述

4.3.3 布尔值数组的方法

  • 在上面这些方法中,布尔值会被强制转换为1(True)和0(False)。因此,sum经常被用来对布尔型数组中的True值计数:
    在这里插入图片描述
  • 另外还有两个方法any和all,它们对布尔型数组非常有用。any用于测试数组中是否存在一个或多个True,而all则检查数组中所有值是否都是True:
    在这里插入图片描述
  • 这两个方法也能用于非布尔型数组,所有非0元素将会被当做True。

4.3.4 排序

  • 和Python的内建列表类型相似,NumPy数组可以使用sort方法按位置排序:
    在这里插入图片描述
  • 你可以在多维数组中根据传递的axis值,沿着轴向每一个一维数据段进行排序:
    在这里插入图片描述
  • 顶级方法np.sort返回的是数组的已排序副本,而就地排序则会修改数组本身。计算数组分位数最简单的办法是对其进行排序,然后选取特定位置的值:
    在这里插入图片描述
  • 更多关于NumPy排序方法以及诸如间接排序之类的高级技术,请参阅附录A。在pandas中还可以找到一些其他跟排序有关的数据操作(比如根据一列或多列对表格型数据进行排序)。

4.3.5 唯一值与其他集合逻辑

  • NumPy包含一些针对一维ndarray的基础集合操作。常用的一个方法是np.unique,返回的是数组中唯一值排序后形成的数组:
    在这里插入图片描述
  • 将np.unique和纯Python实现比较:
    在这里插入图片描述
  • 另一个函数,np.in1d,可以检查一个数组中的值是否在另外一个数组中,并返回一个布尔值数组:
    在这里插入图片描述
  • 表4-6是NumPy中集合函数的列表。
    在这里插入图片描述

4.4 使用数组进行文件输入和输出

  • NumPy能够读写磁盘上的文本数据或二进制数据。这一小节只讨论NumPy的内置二进制格式,因为更多的用户会使用pandas或其它工具加载文本或表格数据(见第6章)。

  • np.save和np.load是读写磁盘数组数据的两个主要函数。默认情况下,数组是以未压缩的原始二进制格式保存在扩展名为.npy的文件中的:

np.save('some_array',arr)
  • 如果文件存放路径中没写.npy时,后缀名会被自动加上。硬盘上的数组可以使用np.load进行载入:
    在这里插入图片描述
  • 你可以使用np.savez并将数组作为参数传递给该函数,用于在未压缩文件中保存多个数组:
np.savez('arr_archive.npz',a=arr,b=arr)
  • 加载.npz文件时,你会得到一个类似字典的对象,该对象会对各个数组进行延迟加载:
    在这里插入图片描述
  • 如果要将数据压缩,可以使用numpy.savez_compressed:
np.savez_compressed('arrays_compressed.npz',a=arr,b=arr)

4.5 线性代数

  • 线性代数(如矩阵乘法、矩阵分解、行列式以及其他方阵数学等)是任何数组库的重要组成部分。不像某些语言(如MATLAB),通过*对两个二维数组相乘得到的是一个元素级的积,而不是一个矩阵点积。因此,NumPy提供了一个用于矩阵乘法的dot函数(既是一个数组方法也是numpy命名空间中的一个函数):
    在这里插入图片描述

  • 一个二维数组和一个长度合适的一维数组之间的矩阵乘积,其结果是一个一维数组:
    在这里插入图片描述

  • 特殊符号@也作为中缀操作符,用于点乘矩阵操作:
    在这里插入图片描述

  • numpy.linalg中有一组标准的矩阵分解运算以及诸如求逆和行列式之类的东西。它们跟MATLAB和R等语言所使用的是相同的行业标准线性代数库,如BLAS、LAPACK、Intel MKL(Math Kernel Library,可能有,取决于你的NumPy版本)等:
    在这里插入图片描述
    在这里插入图片描述

  • 表达式X.T.dot(X)计算X和它的转置X.T的点积。

  • 表4-7中列出了一些最常用的线性代数函数。
    在这里插入图片描述

伪随机数生成

  • numpy.random模块对Python内置的random进行了补充,增加了一些用于高效生成多种概率分布的样本值的函数。例如,你可以用normal来得到一个标准正态分布的4×4样本数组:
    在这里插入图片描述
  • 而Python内置的random模块则只能一次生成一个样本值。从下面的测试结果中可以看出,如果需要产生大量样本值,numpy.random快了不止一个数量级:
    在这里插入图片描述
  • 我们说这些都是伪随机数,是因为它们都是通过算法基于随机数生成器种子,在确定性的条件下生成的。你可以用NumPy的np.random.seed更改随机数生成种子:
np.random.seed(1234)
  • numpy.random的数据生成函数使用了全局的随机种子。要避免全局状态,你可以使用numpy.random.RandomState,创建一个与其它隔离的随机数生成器:
    在这里插入图片描述
  • 表4-8列出了numpy.random中的部分函数。在下一节中,我将给出一些利用这些函数一次性生成大量样本值的范例。
    在这里插入图片描述

示例:随机漫步

  • 我们通过模拟随机漫步来说明如何运用数组运算。先来看一个简单的随机漫步的例子:从0开始,步长1和-1出现的概率相等。

  • 下面是一个通过内置的random模块以纯Python的方式实现1000步的随机漫步:

import random
position = 0
walk = [position]
steps = 1000
for i in range(steps):
    step = 1 if random.randint(0, 1) else -1
    position += step
    walk.append(position)
  • 图4-4是对上面随机漫步的前100步的可视化:
plt.plot(walk[:100])

在这里插入图片描述

  • 不难看出,这其实就是随机漫步中各步的累计和,可以用一个数组运算来实现。因此,我用np.random模块一次性随机产生1000个“掷硬币”结果(即两个数中任选一个),将其分别设置为1或-1,然后计算累计和:
    在这里插入图片描述
  • 由此我们开始从漫步轨道上提取一些统计数据,比如最大值、最小值等:
    在这里插入图片描述
  • 现在来看一个复杂点的统计任务——首次穿越时间,即随机漫步过程中第一次到达某个特定值的时间。假设我们想要知道本次随机漫步需要多久才能距离初始0点至少10步远(任一方向均可)。np.abs(walk)>=10可以得到一个布尔型数组,它表示的是距离是否达到或超过10,而我们想要知道的是第一个10或-10的索引。可以用argmax来解决这个问题,它返回的是该布尔型数组第一个最大值的索引(True就是最大值):
    在这里插入图片描述
  • 请注意,这里使用argmax效率并不高,因为它总是完整地扫描整个数组。在这个特殊的示例中,一旦True被发现,我们就知道最大值了。

4.7.1 一次性模拟多次随机漫步

  • 如果你希望模拟多个随机漫步过程(比如5000个),只需对上面的代码做一点点修改即可生成所有的随机漫步过程。只要给numpy.random的函数传入一个二元元组就可以产生一个二维数组,然后我们就可以一次性计算5000个随机漫步过程(一行一个)的累计和了:
    在这里插入图片描述
  • 现在我们可以计算出这些随机步的最大值和最小值了:
    在这里插入图片描述
  • 得到这些数据之后,我们来计算30或-30的最小穿越时间。这里稍微复杂些,因为不是5000个过程都到达了30。我们可以用any方法来对此进行检查:
    在这里插入图片描述
  • 然后我们利用这个布尔型数组选出那些穿越了30(绝对值)的随机漫步(行),并调用argmax在轴1上获取穿越时间:
    在这里插入图片描述
  • 请尝试用其他分布方式得到漫步数据。只需使用不同的随机数生成函数即可,如normal用于生成指定均值和标准差的正态分布数据:
steps = np.random.normal(loc=0, scale=0.25,
                         size=(nwalks, nsteps))

4.8 本章小结

  • 本书的后续章节的大部分,都会集中在打造pandas数据处理技能上,我们将会继续在基于数据的风格下处理数据。在附录A中,我们将深挖NumPy的特性,帮助你进一步提高数组计算技能。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值