autorenew
Compartiendo código en Elixir

Compartiendo código en Elixir

Cuando uno viene de programar en C#, piensa que utilizar código de otros proyectos o namespaces será igual de sencillo en todos los lenguajes. Pero resulta que hay lenguajes como Elixir en los que la cosa no es tan sencilla y existen varias maneras de compartir código entre módulos. Sin duda es una de las cosas que más me ha costando pillar (igual es que soy un zote), así que he pensado en escribirlo para aclararme yo, y de paso aclarar a quién pueda leerlo. Ojo, no es que sea difícil, pero hay que tener en cuenta que en Elixir existen las macros y que no existe el concepto de namespace. Así que hay más métodos para importar que el clásico using de C#.

Alias

Esta es sin duda la directiva más fácil de entender. Aunque no se utiliza para compartir código, si puede simplificar mucho lo que escribamos. En Elixir, coomo he dicho antes, no existen namespaces, pero se pueden prefijar los módulos para tenerlos organizados. Imaginando el siguiente módulo:

defmodule Console.Writer do
  def write_line(text) do
    IO.puts(text)
  end
end

El módulo es sencillo de entender. Cuando desde otro módulo, queramos escribir algo en la consola, solo tendremos que escribir Console.Writer.write_line "mensaje en pantalla". Si solo lo tenemos que escribir una vez, no hay problema. ¿Y si lo utilizamos muchas veces? Entonces es cuando alias nos puede servir de ayuda. Por ejemplo si al principio de nuestro módulo, escribimos lo siguiente:

alias Console.Writer 

Nos bastará con poner Writer.write_line "mensaje en palntalla", para hacerlo funcionar. Pero si todavía queremos reducir más, o poner otro nombre diferente podemos hacer esto:

alias Console.Writer as: console

Ahora usar nuestro Writer será tan sencillo como escribir console.write_line "mensaje en pantalla"

Import

Con import podemos hacer algo parecido a alias, pero en este caso además importamos las funciones para que podamos utilizarlas directamente en nuestros módulos. Se ve mejor con un ejemplo. Si abrimos iex y directamente tratamos de usar la librería Enum, pasará esto:

iex(1)> sort [2, 1, 5]
** (CompileError) iex:1: undefined function sort/1

Si hacemos Enum.sort [2, 1, 4] el código funcionará, pero como vemos estamos obligados a escribir el nombre completo. En cambio con un import al principio funcionará sin problemas:

iex(1)> import Enum
Enum
iex(2)> sort [2,1,5]
[1, 2, 5]

Ahora ya no hace falta que escribamos toda la ruta de las funciones que hay en Enum (como sort), ya que estas funciones están disponibles en nuestro módulo, como si las hubiéramos programado nosotros.

Con import también podemos indicar qué funciones queremos importar, evitando cargar todas.

import Integer only: :macros #importa solo macros de la librería Integer
import Integer only: :functions #importa solo las funciones de la librería Integer
import Enum only: [sort:1] #importa solo la función sort con "arity" 2 (que recibe dos parámetros) de la librería Enum

Require

En Elixir podemos utilizar macros para ampliar el lenguaje. De hecho Elixir está lleno de macros que podremos usar y aquí es cuando entra la directiva require.

Para que una macro funcione al usarla desde nuestros módulos, le tendremos que decir al compilador que necesitamos tenerla lista y compilada, antes de ejecutar nuestro código. Por ejemplo para utilizar la macro is_even de la librería Integer, antes tendremos que hacer un require.

defmodule Enteros do
  require Integer

  def es_par(num) do
    Integer.is_even(num)
  end
end

Si no lo hacemos Elixir nos devolverá el error “you must require Integer before invoking the macro Integer.is_even/1”. En definitiva, para usar macros, nos tenemos que asegurar de que están cargadas antes de utilziarlas, y esto lo hacemos con require.

Use

En este caso use se trata de una macro, que está relacionada con require. Si utilizamos use en un módulo, nos aseguramos de primero hacer un require y luego llamar a la función __using__ para inicializar algún tipo de estado o importar librerías asociadas. Esta función es una especie de constructor, salvando las distancias.

Un caso típico es el del framework de test que podemos usar en Elixir. Así, cuando hacemos esto:

defmodule AssertionTest do
  use ExUnit.Case, async: true

  test "always pass" do
    assert true
  end
end

El compilador acaba traduciéndolo en esto:

defmodule AssertionTest do
  require ExUnit.Case
  ExUnit.Case.__using__([async: true])

  test "always pass" do
    assert true
  end
end

La función __using__ la podremos definir en cualquiera de nuestros módulos para poder llamarla al utilizar use.

Conclusión

En definitiva, Elixir tiene varias formas diferentes de compartir código entre módulos, y es improtante que las conozcamos todas. Aunque algunas como alias e import son sencillas de entender, use y require son algo más complejas, ya que implican la utilización de macros. Y espero poder hablaros de macros en un futuro no muy lejano.