Saltar al contenido

MicroEntradas: numpy.vectorize

Hoy vamos a ver numpy.vectorize que, como la documentación oficial indica, sirve para ‘vectorizar’ funciones que solo aceptan escalares como entrada. La entrada que podemos meter es una lista de objetos o un ‘numpyarray’ y nos devolverá como resultado un ‘numpyarray’.

No os engañéis, normalmente, cuando hablamos de vectorizar pensamos en varios órdenes de magnitud de mejora en el rendimiento pero eso no es lo que hace esta función :-). Podéis pensar en esta función como algo parecido a usar map.

Por ejemplo, la función abs solo acepta un entero o un float. Por ejemplo, lo siguiente:

Nos devolvería el valor 3. Hasta ahí bien. Pero si queremos hacer lo siguiente:

Nos devolverá un TypeError: bad operand type for abs(): 'list'.

Podríamos usar map para ello pero no obtendríamos un ‘numpyarray’ como salida y en Python 3 devuelve un iterador que para transformarlo a, por ejemplo, una lista o un ‘numpyarray’ debemos añadir un paso más, la conversión a lista (en Python 2 devuelve directamente una lista) o dos pasos, la conversión a lista y esta a ‘numpyarray’.

Vamos a ver cómo podemos ‘vectorizar’ la función abs sin usar map y usando numpy.vectorize (antes deberéis importar numpy como np):

que podemos usar de la siguiente forma:

Et voilà. ya tenemos lo que queríamos. Veamos como va el rendimiento de lo que acabamos de hacer:

La última opción es equivalente a hacer np.array(list(map(abs, kk))) en tiempo. La versión vectorizada, vectabs, es 21 veces más lenta que la que podemos obtener usando numpy.abs por lo que no tendríamos una gran ganancia pero, sin embargo, vemos que es casi tres veces más rápida que la versión usando map (o una ‘list comprehension’) por lo que algo ganariamos respecto a Python 3 puro :-).

Como apuntes finales, si sabéis de alguna función en CPython que no existe equivalente en numpy y la necesitáis usar quizá podéis obtener una ganancia usando numpy.vectorize. Si tenéis que escribir la función vosotros, escribidla pensando en operaciones vectorizadas usando ‘numpyarrays’ y no os hará falta usar numpy.vectorize.

Como punto final. recordad que np.vectorize no es más que un decorador por lo que lo siguiente sería perfectamente válido:

Saludos.

[Editado: corrección de algún bug, disculpas!!]

2 comentarios en «MicroEntradas: numpy.vectorize»

  1. Me ha gustado, algunas veces pensé en esto pero nunca investigué si existía un “vectorizador”.
    Agrego una expresión que falto que asombrosamente le gana hasta a np.abs, usando generadores:

    In [11]: timeit np.abs(kk)
    10 loops, best of 3: 29.1 ms per loop
    In [12]: timeit vect_abs(kk)
    1 loops, best of 3: 387 ms per loop
    In [13]: timeit np.array([abs(i) for i in kk])
    1 loops, best of 3: 1.32 s per loop
    In [14]: timeit np.array(abs(i) for i in kk)
    100000 loops, best of 3: 11.2 µs per loop

    Obligando a timeit a hacer 100 loops me dan los mismo resultados, ¡2600 veces mas rápido que np.abs!

Deja una respuesta

Tu dirección de correo electrónico no será publicada.

ninety three − = 89

Pybonacci