Numerical Python o NumPy

NumPy es una biblioteca para realizar cálculo en el campo del algebra lineal. Originalmente fue derivado de un paquete de rutinas hecho para Fortran, conocido como BLAS.

NumPy es un paquete fundamental para utilizar python en computación de cálculos científicos. Esta es una biblioteca que provee de objetos equivalentes a arreglos multidimensionales y todo un conjunto de rutinas para una operación rápida en los campos de la matemáticas, lógica, manipulación de formas, ordenamientos, entre/salida de datos, Trasformada discreta de Fourier, álgebra lineal básica, operaciones estadísticas básicas, simulaciones con números al azar y otros muchos temas.

Es una biblioteca más importante en el uso científico de python.

Se puede conseguir muchas mas información en www.numpy.org

Numpy en un notebook

Importando NumPy

Como toda biblioteca externa se debe cargar para que pueda ser usada por el notebook. El Python de por si no tiene esta biblioteca en memoria, por lo cual, debe ser el usuario el que decida utilizarla y para ello debe "importarla". Al hacerlo cargará no sólo los algoritmos que lleva programados, sino además las Clases con las cuales se definene las variables propias de la biblioteca. Los objetos que se usan en NumPy son básicametne arreglos que mantienen cierta similitud con los vectores y matrices del álgebra.

Primero tenemos que cargar la biblioteca, es bastante popular llamar "np" a la biblioteca numPy. Aunque no es obligatorio llamarla como "np" ya es un estandard reconocido.

In [1]:
import numpy as np

Este comando carga la biblioteca NumPy, pero con un cambio de nombre, la traducción literal del comando sería "Carga NumPy como np". desde esta manera Numpy de ahora en adelante se llamará np en mi programa. Es normal renombrarla como "np", aunque nada evita que se pudiera haber elegido cualquier otro nombre.

La clase de los arreglos (array)

NumPy usa sus propias variables, es decir como Python es un lenguaje orientado a objetos se definió una clase para generar objetos con las propiedades necesarios para hacer álgebra vectorial. Es decir uso de vectores, matrices, cubos, etc, con mucha eficiencia y respetando propiedades del álgebra. Básicamente NumPy utiliza arreglos que son propios de su biblioteca.

Veamos el uso de la función array( ) que crea objetos NumPy a partir de variables de Python

Puedo crear un arreglo numpy tomándolo de una lista o tupla, o creándolos nuevos.

Veamos el ejemplo de usar una lista y con ella crear un arreglo NUmpy.

Para ello uso la función array de NumPy, y como a NumPy lo llamé "np", a la función la tengo que llamar np.array( )

El arreglo NumPy creado ya no es una lista (aunque se le parece mucho), tiene otras propiedades.

In [2]:
a_lista = [1,2,3,4,5,6]

a = np.array(a_lista)

print(a)
[1 2 3 4 5 6]

Notar que ahora no tenemos las "," típicas de una lista.

O podría hacer directamente

In [3]:
a = np.array([1,2,3,4,5,6])

Si pregunto el tipo de variable que tengo ahora

In [4]:
print("Es de la clase:", type(a))
Es de la clase: <class 'numpy.ndarray'>
In [5]:
# funciona también con tuplas:

b = np.array((1,2,3))

print(b)
print("Es de la clase:",type(b)) # Ya NO es una Tupla.... y además 
                                 # desaparecen las comas de la lista o tupla
[1 2 3]
Es de la clase: <class 'numpy.ndarray'>
In [6]:
# Creo una lista
L = [1, 2, 3, 4]
# Convierto mi lista a un arreglo NumPy 
a = np.array(L)

print("La lista L es del tipo: ",type(L))
print("a es de tipo:",type(a))
print("El arreglo a es del tipo:",a.dtype)
print("")
print(L)
print(a)
La lista L es del tipo:  <class 'list'>
a es de tipo: <class 'numpy.ndarray'>
El arreglo a es del tipo: int64

[1, 2, 3, 4]
[1 2 3 4]

El dtype es un comando de Numpy. Me indica que el arreglo "a" no es una lista de Python. dtype me indica el tipo de arreglo dentro de las posibilidades de arreglos Numpy. En este caso es tipo Integer*8 (64 bits).

A diferencia de las listas los arreglos Numpy engloban a variables que son del mismo tipo. Es decir que todo el arreglo Numpy es del mismo tipo de variable (Real*8, Int*8, complejo, etc)

Una vez definido, todos los agregados se modifican para que sean válidos en el tipo de arreglo final.

In [7]:
Lista2=[1,2,22.] # <-- 1 y 2 enteros, 22. es real 
Lista_numpy=np.array(Lista2)

print(Lista_numpy)
[ 1.  2. 22.]

Todo el arreglo terminó siendo real.


Los arreglos NumPy son estructuras donde los cálculos son rápidos y eficientes.

Para ver la ventaja real de esta biblioteca usaremos el comando del notebook %timeit (que es una función del notebook, no es una orden de Python) que me permite tomar tiempos.

range es una función de python.

arange es una función que se agrega porque viene con numPy. Esta última crea los mismos valores de la lista, pero ahora en array NumPy

In [8]:
L = range(1000)  # Crea una lista

%timeit for L in range(1000):    A=L**2

print('Pero usando NumPy')

A = np.arange(1000)  # Crea un arreglo Numpy

%timeit A2 = A**2
191 µs ± 3.78 µs per loop (mean ± std. dev. of 7 runs, 10,000 loops each)
Pero usando NumPy
944 ns ± 14.9 ns per loop (mean ± std. dev. of 7 runs, 1,000,000 loops each)

Moraleja 1:

La idea que hay que tener presente cuando se usa NumPy, es que se va a trabajar con arreglos como variables. No hay que pensar en escribir código que apunte a algoritmos donde se trabaja con los elementos de a uno en un arreglo. El código se escribe pensando las operaciones matemáticas que se realizan sobre vectores y matrices. Veremos esto con detalle durante el curso.

Los arreglos NumPy llevan consigo al ser objetos, atributos y funciones. Estas funciones en particular son mucho más eficientes que las originales de Python. Probemos creando un arreglo usar primero la función que busca el mínimo valor de Python (min(a)) y luego lo que hace el mismo trabajo pero con la función que reside en el objeto NumPy (a.min()). Noten que tiene el mismo nombre pero que se las llama para su uso de forma diferente.

In [9]:
# construyo un vector de 10.000.000 números con primer valor 1000 y último 0, con paso -0.00001

a = np.arange(1000,0,-0.0001)

# veo que tengo en a
print("Largo de a: ",len(a))

%timeit min(a)
%timeit a.min()
Largo de a:  10000000
552 ms ± 25 ms per loop (mean ± std. dev. of 7 runs, 1 loop each)
3.23 ms ± 66.3 µs per loop (mean ± std. dev. of 7 runs, 100 loops each)

Moraleja 2:

Hay que usar las funciones que vienen en las biblioteca que son especifícas para un tema particular, en el caso de álgebra vectorial (como en este último ejemplo) es prácticamente obligatorio usar las funciones de los objetos NumPy.

Pueden ser de varias dimensiones 1D, 2D, 3D, ...

In [10]:
a = np.array([1,2,3,4,5,6])
b = np.array([[1,2],[1,4]])
c=np.array([[[1,2], [2,3]], [[3,4], [4,5]]])

print("Imprimo el atributo shape")
print( a.shape, b.shape, c.shape)
print("")
print("Pero si me fijo el len()")
print(len(a),len(b), len(c))
Imprimo el atributo shape
(6,) (2, 2) (2, 2, 2)

Pero si me fijo el len()
6 2 2

El atributo shape me cuenta de las dimensiones del arreglo, en cambio la función len( ) me cuenta de su largo.

El comando shape me devuelve una tupla con la información, en cambio len( ) me devuelve un número.

Si tengo varias dimensiones y quiero saber la cantidad total de elementos es conveniente imprimir el atributo size.

Veamos como imprimimos el array c y sus componentes ya que este arreglo es trimensional.

In [11]:
print(c[0])
[[1 2]
 [2 3]]
In [12]:
print(c[0][0])
[1 2]
In [13]:
print(c[0][0][0])  #<-- recién aquí me devuelve un número
1
In [14]:
# size me da la cantidad de valores en el array
b.size
Out[14]:
4

Pero estrictamente la dimensión me la da el atributo ndim

In [15]:
print (a.ndim, b.ndim, c.ndim )
1 2 3

La forma como NumPy numera los ejes, se conoce como el estilo "C" (por el lenguaje C que lo hace de esa manera) y es diferente al estilo Fortran. Aunque siempre es más cómodo visualizar las distintas dimensiones como arreglos de arreglos de otros arreglos...

dimensiones

Crédito del dibujo:https://www.tomasbeuzen.com/python-programming-for-data-science/chapters/chapter5-numpy.html


La biblioteca numPy trae funciones asociadas.

Las puedo buscar en el manual (www.numpy.org), veamos algunos ejemeplos:

In [16]:
a = np.array([1,2,3,4,5,6,7])

Por ejemplo, probemos con dos, la que calcula promedios de un array a.mean( ) y la que me encuentra el valor máximo a.max( )

In [17]:
print(a.mean(), a.max())
4.0 7

mean( ) y max( ) son métodos (funciones) de la clase del arreglo, necesitan los paréntesis ( ).

In [18]:
print(a.mean) # Esto así imprime de que trata la funcion no su resultado
<built-in method mean of numpy.ndarray object at 0x7fb7d07a3c90>
In [19]:
print ("Imprimo b: ",b)
print ("Imprimo el promedio de b: ",b.mean()) # Promedio sobre todo el arreglo 
print ("Imprimo el promedio de las columnas de b: ",b.mean(axis=0)) # Promedio sobre el primer eje (columnas)
print ("Imprimo el promedio de las filas de b: ", b.mean(1)) # Promedio sobre las filas, se sobreentiende que es "axis=1"
Imprimo b:  [[1 2]
 [1 4]]
Imprimo el promedio de b:  2.0
Imprimo el promedio de las columnas de b:  [1. 3.]
Imprimo el promedio de las filas de b:  [1.5 2.5]

Creando arreglos usando sentencias específicas

La idea es que el arreglo reciba una cantidad de elementos creados en una forma automática y que cumplen reglas que se especifican como bucles. Es decir con un inicio, un final y un paso. Donde este último no tiene que ser obligtoriamente lineal. Por ejemplo, podríamos tener un paso geométrico, logarítmico, exponencial, etc.

También veremos que existen órdenes que copian la forma de otro arreglo, pero no sus elementos, sólo su forma, es decir su tamaño y dimensión.

In [20]:
print(np.arange(10))
[0 1 2 3 4 5 6 7 8 9]
In [21]:
print(np.linspace(0, 1, 10)) # Comienzo, final, y número de puntos 
[0.         0.11111111 0.22222222 0.33333333 0.44444444 0.55555556
 0.66666667 0.77777778 0.88888889 1.        ]
In [22]:
print(np.linspace(0, 1, 10, endpoint=False)) # No incluimos el punto final, o lo hacemos hasta 11. 
[0.  0.1 0.2 0.3 0.4 0.5 0.6 0.7 0.8 0.9]
In [23]:
print(np.zeros(2)) # Lleno con ceros
[0. 0.]
In [24]:
print(np.zeros((2,2))) # es un arreglo de dos dimensiones lleno con ceros 
[[0. 0.]
 [0. 0.]]
In [25]:
print(np.ones_like(a)) # Esto es muy bueno, creo un arreglo (o tupla) 
                       # usando las propiedades de otro que ya tengo
[1 1 1 1 1 1 1]
In [26]:
print(np.zeros_like(a)+3) # Me sirve para llenar con otro valor que no son ceros o 1's
[3 3 3 3 3 3 3]
In [27]:
print(np.ones_like([1,2,3]))
[1 1 1]

Puedo crear arreglos vacíos de números pero con la forma final que deben tomar.

In [28]:
a22=np.empty([2, 2])
a11=np.empty([10])
In [29]:
print(np.logspace(0, 2, 10)) # va de 10**comienzo hasta 10**final y calculo 10 valores 
[  1.           1.66810054   2.7825594    4.64158883   7.74263683
  12.91549665  21.5443469   35.93813664  59.94842503 100.        ]

Una manera muy típica de trabajar en python es primero crea el array como unidimensional y luego darle forma con el comando reshape( ), que lo vimos en Fortran 90.

In [30]:
a = np.array([1,2,3,4,5,6])
b = a.reshape((3,2)) # Este comando no cambia la "forma" de  a
print(a)
print('')
print(b)
[1 2 3 4 5 6]

[[1 2]
 [3 4]
 [5 6]]

Mientras que la función ravel( ) lo vuelve unidimensional.

In [31]:
print(b.ravel())
[1 2 3 4 5 6]

OJO - CUIDADO, los arreglos son objetos y cuando se reasignan no se copian, entonces apuntan a la misma memoria

In [32]:
b = a.reshape((3,2))
c = a.reshape((3,2))
print(a.shape, b.shape)
(6,) (3, 2)
In [33]:
print(b)
print("")
b[1,1] = 100 # modifico un valor del arreglo 
print(b)
[[1 2]
 [3 4]
 [5 6]]

[[  1   2]
 [  3 100]
 [  5   6]]
In [34]:
print(a) # !!! a y b son el mismo objeto comparten el mismo espacio de memoria, 
         # es decir apuntan a los mismos valores 
 
[  1   2   3 100   5   6]
In [35]:
print(b[1,1],a[3]) # el mismo valor
100 100
In [36]:
c = a.reshape((2,3)).copy() # Esta es la solución
In [37]:
print(a)
print('')
print(c)
[  1   2   3 100   5   6]

[[  1   2   3]
 [100   5   6]]
In [38]:
c[0,0] = 8888
print(a)
print(c)
[  1   2   3 100   5   6]
[[8888    2    3]
 [ 100    5    6]]

Números al azar (random) en Numpy

Tengo funciones específicas ya programadas en NumPy para realizar la tarea.

In [39]:
ran_uniform = np.random.rand(5) # Entre 0 y 1
ran_normal = np.random.randn(5) # Gausiana promedio 0 varianza 1
print("Números con distribución Uniforme: ",ran_uniform)
print ('')
print ("Números con distribución Gaussiana: ",ran_normal)
print ('')
ran_normal_2D = np.random.randn(5,5) # Gausiana promedio 0 varianza 1
print("Números con distribución Gaussiana, pero ahora en una matriz: \n\n",ran_normal_2D)
Números con distribución Uniforme:  [0.08463704 0.72696046 0.54601244 0.34173924 0.66038532]

Números con distribución Gaussiana:  [ 2.05745191e-03 -1.38662479e+00  2.32558371e+00  2.00297588e+00
 -7.71178641e-01]

Números con distribución Gaussiana, pero ahora en una matriz: 

 [[-1.21818555  0.06901503  0.8052623  -0.62921366  1.52904545]
 [ 0.00648282 -1.37691634 -0.1438621   0.46809425  0.2798729 ]
 [-0.31582644  1.00920015  0.26261918 -1.36323256  0.79347144]
 [ 1.05698573 -0.79930771  1.29869682 -0.48309304  1.87625487]
 [-0.4909549  -0.02271899 -0.21362269  2.15259979 -0.60165675]]

Si en la celda siguiente elimino el # en la línea y corro la celda me da como resultado un "help" del comando. Estas ayudas que se agregan en las funciones son los comentarios que se agregaron en el código de la función como docstrings.

In [40]:
#np.random.randn?

Slicing (cortando fetas)

Es tal como hemos visto con los comandos originales de Python. El rabanado de un objetos de Numpy se hace con los mismos comandos.

In [41]:
a = np.arange(10)
print(a)
[0 1 2 3 4 5 6 7 8 9]
In [42]:
print(a[1:8:3])
[1 4 7]

Usando arreglos y máscaras

Puedo en NumPy cambiar varios números en una sólo línea con un lista de estos elementos. Veamos un ejemplo:

In [43]:
print(a)
a[[2,4,6]] = -999   # Puse como ínidices en el arreglo a, una lista de elementos 2,4,6
print(a)
[0 1 2 3 4 5 6 7 8 9]
[   0    1 -999    3 -999    5 -999    7    8    9]

Veamos como construir un arreglo NumPy donde sus elementos son variables lógicas

In [44]:
a = np.random.randint(0, 100, 20) # min, max, N
print(a)
[42 99  8 80 38 14 18 60 69  1 42 52 33 25 71 23  2 81 16 62]
In [45]:
a < 50
Out[45]:
array([ True, False,  True, False,  True,  True,  True, False, False,
        True,  True, False,  True,  True, False,  True,  True, False,
        True, False])

La respuesta al operador de relación fue un arreglo con valores booleanos, es decir es la respueta si es verdadera o falsa la pregunta que se realizó en cada elemento.

Veamos esto con más detalle y como usarlo en un caso real:

Operaciones con arreglos - Filtrando con máscaras

Utilizo el arreglo a del ejemplo anterior:

In [46]:
a 
Out[46]:
array([42, 99,  8, 80, 38, 14, 18, 60, 69,  1, 42, 52, 33, 25, 71, 23,  2,
       81, 16, 62])

Y lo uso para realizar operacion matemáticas simples:

In [47]:
a + 1
Out[47]:
array([ 43, 100,   9,  81,  39,  15,  19,  61,  70,   2,  43,  53,  34,
        26,  72,  24,   3,  82,  17,  63])

Hay que recordar que ahora la variable para los cálculos es todo el arreglo. Y podemos hacer operaciones bastante más complejas, como la que sigue:

In [48]:
a**2 + 3*a**3
Out[48]:
array([ 224028, 2920698,    1600, 1542400,  166060,    8428,   17820,
        651600,  990288,       4,  224028,  424528,  108900,   47500,
       1078774,   37030,      28, 1600884,   12544,  718828])

Creo un arreglo más pequeño:

In [49]:
a = np.arange(10)
print("a es:",a)
a es: [0 1 2 3 4 5 6 7 8 9]

Y realizo una operación más compleja y asigno el resultado a otro arreglo que llamo "b"

In [50]:
b = a**2 + (a+1)**2
print("b es:",b)
b es: [  1   5  13  25  41  61  85 113 145 181]

Hago otra operación a la que asigno el resultado al arreglo "c"

In [51]:
c = (a+2)**2
In [52]:
print("b es:",b)
print("c es:",c)
b es: [  1   5  13  25  41  61  85 113 145 181]
c es: [  4   9  16  25  36  49  64  81 100 121]

Podría entonces preguntar que valores están repetidos en el mismo lugar en los arreglos b y c

In [53]:
print(b == c)
[False False False  True False False False False False False]

Guardo el resultado en el arreglo "máscara"

In [54]:
mascara = b==c
print("mascara es:", mascara)
mascara es: [False False False  True False False False False False False]

¿Y si la máscara la uso para filtrar el vector? ¿Cómo funciona eso?

Es decir pido que imprima b[mascara], pero mascara ahora funciona como un arreglo de los índices de b al poner de esa manera el comando. Sólo pasarán el filtro los elementos verdaderos.

In [55]:
print("b es:",b)
print("b filtrado por la mascara es: ",b[mascara])
b es: [  1   5  13  25  41  61  85 113 145 181]
b filtrado por la mascara es:  [25]

También podría haber usado un expresión más compleja

In [56]:
mascara= b < 15 
print(mascara)
[ True  True  True False False False False False False False]

Podría entonces usar el arreglo máscara para filtrar los valores que cumplen con la condición solicitada.

In [57]:
d=b[mascara]
print(d)
[ 1  5 13]

Incluso podría haber hecho todo esta operación en un sólo comando:

In [58]:
print(b>c)
print(a[b>c])
[False False False False  True  True  True  True  True  True]
[4 5 6 7 8 9]

Cuidado que esto último lo puedo hacer porque a,b y c tienen el mismo tamaño y los elementos se corresponden en la numeración

 NumPy maneja casi todas las expresiones matemáticas, log, trigonométricas, etc

In [59]:
a = np.arange(18)
print("a es:",a)
print("")
print("El log10 de a es:",np.log10(a))
a es: [ 0  1  2  3  4  5  6  7  8  9 10 11 12 13 14 15 16 17]

El log10 de a es: [      -inf 0.         0.30103    0.47712125 0.60205999 0.69897
 0.77815125 0.84509804 0.90308999 0.95424251 1.         1.04139269
 1.07918125 1.11394335 1.14612804 1.17609126 1.20411998 1.23044892]
/var/folders/3c/tj_0cb8x1xd_hwlkqgz3q1lr0000gn/T/ipykernel_53928/3913186439.py:4: RuntimeWarning: divide by zero encountered in log10
  print("El log10 de a es:",np.log10(a))

Notar que hubo un error al sacar el log10 de 0. Pero el programa siguió corriendo.

Ese error fue notificado en el cartel que sale con el "RuntimeWarning".

Podemos encontrarnos elementos que dieron errores y estos pueden ser: -inf, +inf, y NaN.

Operaciones con vectores y matrices

In [60]:
a = [[1, 0], [0, 1]]
b = [[4, 1], [2, 2]]

c = np.matmul(a, b)

print("c es:",c)
c es: [[4 1]
 [2 2]]
In [61]:
a = [[1, 0], [0, 1]]
b = [1, 2]
c= np.matmul(a, b)

print(c)

c= np.matmul(b, a)
print(c)
[1 2]
[1 2]
In [62]:
a = np.arange(1000000).reshape(1000,1000)
b = np.ones_like(a)
print(a)
print("")
print(b)
[[     0      1      2 ...    997    998    999]
 [  1000   1001   1002 ...   1997   1998   1999]
 [  2000   2001   2002 ...   2997   2998   2999]
 ...
 [997000 997001 997002 ... 997997 997998 997999]
 [998000 998001 998002 ... 998997 998998 998999]
 [999000 999001 999002 ... 999997 999998 999999]]

[[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 1 1 ... 1 1 1]
 [1 1 1 ... 1 1 1]]
In [63]:
c=np.matmul(a,b)
print(c)
[[   499500    499500    499500 ...    499500    499500    499500]
 [  1499500   1499500   1499500 ...   1499500   1499500   1499500]
 [  2499500   2499500   2499500 ...   2499500   2499500   2499500]
 ...
 [997499500 997499500 997499500 ... 997499500 997499500 997499500]
 [998499500 998499500 998499500 ... 998499500 998499500 998499500]
 [999499500 999499500 999499500 ... 999499500 999499500 999499500]]

Solución de un sistema lineal de ecuaciones

A = floor(random.rand(4000,4000)*20-10) <-- genera una matriz de 4000 x 4000 valores llena con números al azar

In [64]:
A = np.random.rand(4000,4000)
print("Imprimo A")
print (A)

b = np.floor(np.random.rand(4000,1)*20-10)
print("")
print("Imprimo b")
print(b)

# Resolvamos Ax = b usando el comando NumPy para la tarea: 
x = np.linalg.solve(A,b)

print("")
print("Imprimo el resultado")
print(x)
Imprimo A
[[0.34457515 0.90622766 0.51743853 ... 0.9518204  0.49172172 0.75930953]
 [0.64704686 0.33846378 0.14012165 ... 0.1920994  0.24423514 0.52709835]
 [0.6845568  0.64685182 0.16962833 ... 0.68522664 0.41829059 0.23665596]
 ...
 [0.28821327 0.59142155 0.30549559 ... 0.57754229 0.81541611 0.06348346]
 [0.76164222 0.18046949 0.98859954 ... 0.46425124 0.60351001 0.90565223]
 [0.14555494 0.69397486 0.78967248 ... 0.51054438 0.66363044 0.01982547]]

Imprimo b
[[ -3.]
 [ -7.]
 [  0.]
 ...
 [  9.]
 [-10.]
 [ -8.]]

Imprimo el resultado
[[ 27.06116911]
 [-62.02367806]
 [-35.20452622]
 ...
 [ 24.27629025]
 [ 44.88559603]
 [-79.49353748]]

Broadcasting

El broadcasting es la manera que tiene Python de compensar operaciones elemento a elemento entre arreglos de diferentes tamaños.
La situación más simple es cuando se realiza una operación entre un arreglo por un escalar. El escalar en virtualmente repetido todas las veces necesarias para repetir el tamaño del escalar.

Se suele considerar que el broadcasting es un fuerte consumidor de memoria de la computadora.

In [65]:
a = np.array([1.0, 2.0, 3.0])
b = 2.0
a * b
Out[65]:
array([2., 4., 6.])

En este caso hubiese sido similar a que conviertiera b en b=np.array([2.0, 2.0, 2.0])

In [66]:
a = np.array([1.0, 2.0, 3.0])
b = b=np.array([2.0, 2.0, 2.0])
a * b
Out[66]:
array([2., 4., 6.])
In [67]:
a = np.array([[0.0], [10.0], [20.0], [30.0]])
b = np.array([1.0, 2.0, 3.0])
a+b
Out[67]:
array([[ 1.,  2.,  3.],
       [11., 12., 13.],
       [21., 22., 23.],
       [31., 32., 33.]])

Arreglos

Moraleja: Los arreglos sirven para hacer álgebra vectorial, pero NO son vectores, ni matrices...

No tienen las mismas propiedades...

El broadcast sólo se puede realizar si:

  • Los arreglos son iguales o
  • uno de ellos tiene dimensión 1 arreglos son iguales

Forma de hacer las cosas con NumPy

Veamos como hay que razonar para usar NumPy y de paso calculemos PI (una vez más...)

Esta vez usando series de Taylor. En este caso arctan(1) como hizo en su momento Leibniz

$$\frac{\pi}{4} = \sum_{k=0}^{\infty} \frac{(-1)^k}{2k+1}$$

Veamos un la forma de plantearlo en Python, y usando NumPy:

Por lo que calculo la serie de dos maneras, primero estilo Fortran: término por término de la serie mientras los sumo.

La segunda manera en modo NumPy, todos los términos al mismo tiempo, creando un vector de términos y que luego sumo.

La función pow(x, y) is igual a hacer el cálculo $x^y$, y funciona mejor que $x**y$, porque permite valores fracionarios y negativos de $y$.

In [68]:
# Estilo Fortran
n = 10000000
total = 0
for k in range(n):
    total = total + pow(-1,k)/(2*k+1.0)  
print("Cálculo un elemento y lo voy sumando              :",total*4)
Cálculo un elemento y lo voy sumando              : 3.1415925535897915
In [69]:
#Estilo NumPy
k=np.arange(n)
term=pow(-1,k)/(2*k+1.0)
total=term.sum()
print("Todos los términos al mismo tiempo y sumo al final:", total*4)
Todos los términos al mismo tiempo y sumo al final: 3.1415925535897977

Noten que la forma NumPy es mucho más veloz.

Resumen de lo importante para usar NumPy

  • Python con NumPy es muy bueno, pero hay que pensar en términos de matrices (o arreglos). Nadie usa Python sin usar NumPy en el área científica.
  • Para usar NumPy en forma eficiente y que valga la pena, es necesario que al algoritmo a programar sea posible describirlo en términos de álgebra vectorial.
  • Puede paralelizarse para soportar un hardware específico (GPUs o TPUs) en varias encarnaciones, aunque aún no hay una versión oficial.
  • Queda claro que como existen una cantidad importante de órdenes, uno sólo necesita aprender los comandos básicos y para alguna tarea muy específica hay leer la documentación primero para verificar si el comando ya existe.
  • En soporte a la idea anterior, existen demasiados comandos, puede ser una mala idea pensar que se pueden aprender todos.
  • Se siguen hoy en día agregando y modificando comandos. A NumPy aún se lo continúa construyéndo.