NumPy is the fundamental package for scientific computing in Python. It is a Python library that provides a multidimensional array object, various derived objects (such as masked arrays and matrices), and an assortment of routines for fast operations on arrays, including mathematical, logical, shape manipulation, sorting, selecting, I/O, discrete Fourier transforms, basic linear algebra, basic statistical operations, random simulation and much more.
import numpy as npnp.__version__
'2.4.1'
Arrays
In general NumPy arrays are constructed from sequences (e.g. lists), nesting as necessary for the number of desired dimensions.
np.array([1,2,3])
array([1, 2, 3])
np.array([[1,2],[3,4]])
array([[1, 2],
[3, 4]])
np.array([[[1,2],[3,4]], [[5,6],[7,8]]])
array([[[1, 2],
[3, 4]],
[[5, 6],
[7, 8]]])
np.array([1.0, 2.5, np.pi])
array([1. , 2.5 , 3.14159265])
np.array([[True], [False]])
array([[ True],
[False]])
np.array(["abc", "def"])
array(['abc', 'def'], dtype='<U3')
Note that NumPy stores data in row major order, unlike R which used column major order.
Some properties of NumPy arrays:
Arrays have a fixed size at creation
All data must be homogeneous (i.e. consistent type)
Built to support vectorized operations (avoid loops)
Avoids copying whenever possible (inplace operations)
dtype
Calling type() on any NumPy array returns numpy.ndarray - the specific type stored in the array is recorded as the array’s dtype. This is accessible via the dtype attribute and can be set at creation using the dtype argument.
np.array([1,1]).dtype
dtype('int64')
np.array([1.1, 2.2]).dtype
dtype('float64')
np.array([True, False]).dtype
dtype('bool')
np.array([3.14159, 2.33333], dtype = np.double)
array([3.14159, 2.33333])
np.array([3.14159, 2.33333], dtype = np.float16)
array([3.14 , 2.334], dtype=float16)
np.array([1,2,3], dtype = np.uint8)
array([1, 2, 3], dtype=uint8)
dtypes and overflow
Some types have a maximum and/or minimum value that can be stored in them.
If you try to create an array with a value outside of this range you will get an overflow error.
np.array([-1, 1,2], dtype = np.uint8)
OverflowError: Python integer -1 out of bounds for uint8
np.array([1,2,1000], dtype = np.uint8)
OverflowError: Python integer 1000 out of bounds for uint8
If you are instead coercing values using astype() you will not get this error.
np.array([-1, 1,2,1000]).astype(np.uint8)
array([255, 1, 2, 232], dtype=uint8)
Creating 1d arrays
Some common functions and methods for creating 1d arrays:
Arrays are subsetted using the standard python syntax with either indexes or slices, dimensions are separated by commas.
x = np.array([[1,2,3],[4,5,6],[7,8,9]])x
array([[1, 2, 3],
[4, 5, 6],
[7, 8, 9]])
x[0]
array([1, 2, 3])
x[0,0]
np.int64(1)
x[0][0]
np.int64(1)
x[0:3:2, :]
array([[1, 2, 3],
[7, 8, 9]])
x[0:3:2, :]
array([[1, 2, 3],
[7, 8, 9]])
x[0:3:2, ]
array([[1, 2, 3],
[7, 8, 9]])
x[1:, ::-1]
array([[6, 5, 4],
[9, 8, 7]])
Views and copies
Basic subsetting of ndarray objects does not result in a new object, but instead a “view” of the original object. There are a couple of ways that we can investigate this behavior,
Most of the subsetting approaches we’ve just seen can also be used for assignment, just keep in mind that we cannot change the size or type of the ndarray,
x = np.arange(9).reshape((3,3)); x
array([[0, 1, 2],
[3, 4, 5],
[6, 7, 8]])
x[0,0] =-1; x
array([[-1, 1, 2],
[ 3, 4, 5],
[ 6, 7, 8]])
x[0, :] =-2; x
array([[-2, -2, -2],
[ 3, 4, 5],
[ 6, 7, 8]])
x[0:2,1:3] =-3; x
array([[-2, -3, -3],
[ 3, -3, -3],
[ 6, 7, 8]])
x[(0,1,2), (0,1,2)] =-4; x
array([[-4, -3, -3],
[ 3, -4, -3],
[ 6, 7, -4]])
x[0,0] ="A"
ValueError: invalid literal for int() with base 10: 'A'
Reshaping arrays
The dimensions of an array can be retrieved via the shape attribute, these values can be changed via the reshape() method or updating shape
x = np.arange(6); x
array([0, 1, 2, 3, 4, 5])
y = x.reshape((2,3)); y
array([[0, 1, 2],
[3, 4, 5]])
x
array([0, 1, 2, 3, 4, 5])
np.shares_memory(x,y)
True
z = xz.shape = (2,3); z
array([[0, 1, 2],
[3, 4, 5]])
x
array([[0, 1, 2],
[3, 4, 5]])
np.shares_memory(x,z)
True
Implicit dimensions
When reshaping an array, the value -1 can be used to automatically calculate a dimension,
x = np.arange(6); x
array([0, 1, 2, 3, 4, 5])
x.reshape((2,-1))
array([[0, 1, 2],
[3, 4, 5]])
x.reshape((-1,3,2))
array([[[0, 1],
[2, 3],
[4, 5]]])
x.reshape(-1)
array([0, 1, 2, 3, 4, 5])
x.reshape((-1,4))
ValueError: cannot reshape array of size 6 into shape (4)
Flattening arrays
We just saw one of the more common approaches to creating a flat view of an array (reshape(-1)), there are two other common methods / functions:
ravel creates a flattened view of the array and
flatten creates a flattened copy of the array.
w = np.arange(6).reshape((2,3)); w
array([[0, 1, 2],
[3, 4, 5]])
x = w.reshape(-1)x
array([0, 1, 2, 3, 4, 5])
np.shares_memory(w,x)
True
y = w.ravel()y
array([0, 1, 2, 3, 4, 5])
np.shares_memory(w,y)
True
z = w.flatten()z
array([0, 1, 2, 3, 4, 5])
np.shares_memory(w,z)
False
Resizing
The size of an array cannot be changed but a new array with a different size can be created from an existing array via the resize function and method. Note these have different behaviors around what values the new entries will have.
x = np.resize( np.ones((2,2)), (3,3))x
array([[1., 1., 1.],
[1., 1., 1.],
[1., 1., 1.]])
y = np.ones( (2,2)).resize( (3,3))y
Why didn’t this 2nd version work?
Joining arrays
concatenate() is a general purpose function for joining arrays, with specialized versions vstack(), hstack(), and dstack() for rows, columns, and slices respectively.
x = np.arange(4).reshape((2,2)); x
array([[0, 1],
[2, 3]])
y = np.arange(4,8).reshape((2,2)); y
array([[4, 5],
[6, 7]])
np.concatenate((x,y), axis=0)
array([[0, 1],
[2, 3],
[4, 5],
[6, 7]])
np.concatenate((x,y), axis=1)
array([[0, 1, 4, 5],
[2, 3, 6, 7]])
np.vstack((x,y))
array([[0, 1],
[2, 3],
[4, 5],
[6, 7]])
np.hstack((x,y))
array([[0, 1, 4, 5],
[2, 3, 6, 7]])
Joining arrays (cont.)
np.concatenate((x,y), axis=2)
numpy.exceptions.AxisError: axis 2 is out of bounds for array of dimension 2
np.concatenate((x,y), axis=None)
array([0, 1, 2, 3, 4, 5, 6, 7])
np.dstack((x,y))
array([[[0, 4],
[1, 5]],
[[2, 6],
[3, 7]]])
NumPy numerics
Basic operators
All of the basic mathematical operators in Python are implemented for arrays, they are applied element-wise to the array values.
np.arange(3) + np.arange(3)
array([0, 2, 4])
np.arange(3) - np.arange(3)
array([0, 0, 0])
np.arange(3) +2
array([2, 3, 4])
np.arange(3) * np.arange(3)
array([0, 1, 4])
np.arange(1,4) / np.arange(1,4)
array([1., 1., 1.])
np.arange(3) *3
array([0, 3, 6])
np.full((2,2), 2) ** np.arange(4).reshape((2,2))
array([[1, 2],
[4, 8]])
np.full((2,2), 2) ** np.arange(4)
ValueError: operands could not be broadcast together with shapes (2,2) (4,)
Mathematical functions
NumPy provides a wide variety of basic mathematical functions that are vectorized, in general they will be faster than their base equivalents (e.g. np.sum() vs sum()),
np.sum(np.arange(1000))
np.int64(499500)
np.cumsum(np.arange(10))
array([ 0, 1, 3, 6, 10, 15, 21, 28, 36, 45])
np.log10(np.arange(1,4))
array([0. , 0.30103 , 0.47712125])
np.median(np.arange(10))
np.float64(4.5)
The axis parameter
Many aggregate functions accept an axis parameter that specifies which dimension to operate along. axis=0 operates along rows (collapsing rows), axis=1 along columns, etc. Using axis=None (the default) operates on the flattened array.
NumPy has another submodule called random for functions used to generate random values.
In order to use this, you construct a generator via default_rng(), with or without a seed, and then use the generator’s methods to obtain your desired random values.
Advanced indexing is triggered when the selection object, obj, is a non-tuple sequence object, an ndarray (of data type integer or bool), or a tuple with at least one sequence object or ndarray (of data type integer or bool).
There are two types of advanced indexing: integer and Boolean.
Advanced indexing always returns a copy of the data (contrast with basic slicing that returns a view).
Integer array subsetting (lists)
Lists of integers can be used to subset in the same way:
If we want to use a logical operators on an array we need to use &, |, and ~ instead of and, or, and not respectively.
x = np.arange(6); x
array([0, 1, 2, 3, 4, 5])
y = (x %2==0); y
array([ True, False, True, False, True, False])
~y
array([False, True, False, True, False, True])
y & (x >3)
array([False, False, False, False, True, False])
y | (x >3)
array([ True, False, True, False, True, True])
np.where()
np.where() is a vectorized conditional selection function. With three arguments it acts like an element-wise if-else, returning values from one of two arrays based on a condition.
We will now use this to attempt a simple brute force approach to numerical optimization, define a grid of points using meshgrid() to approximate the minimum of the following function:
\[
f(x,y) = (1-x)^2 + 100(y-x^2)^2
\] Considering values of \(x,y \in (-1,3)\), which value(s) of \(x,y\) minimize this function?