View Source Anonymous functions
Anonymous functions allow us to store and pass executable code around as if it was an integer or a string. Let's learn more.
Defining anonymous functions
Anonymous functions in Elixir are delimited by the keywords fn
and end
:
iex> add = fn a, b -> a + b end
#Function<12.71889879/2 in :erl_eval.expr/5>
iex> add.(1, 2)
3
iex> is_function(add)
true
In the example above, we defined an anonymous function that receives two arguments, a
and b
, and returns the result of a + b
. The arguments are always on the left-hand side of ->
and the code to be executed on the right-hand side. The anonymous function is stored in the variable add
.
We can invoke anonymous functions by passing arguments to it. Note that a dot (.
) between the variable and parentheses is required to invoke an anonymous function. The dot makes it clear when you are calling an anonymous function, stored in the variable add
, opposed to a function named add/2
. For example, if you have an anonymous function stored in the variable is_atom
, there is no ambiguity between is_atom.(:foo)
and is_atom(:foo)
. If both used the same is_atom(:foo)
syntax, the only way to know the actual behaviour of is_atom(:foo)
would be by scanning all code thus far for a possible definition of the is_atom
variable. This scanning hurts maintainability as it requires developers to track additional context in their head when reading and writing code.
Anonymous functions in Elixir are also identified by the number of arguments they receive. We can check if a function is of any given arity by using is_function/2
:
# check if add is a function that expects exactly 2 arguments
iex> is_function(add, 2)
true
# check if add is a function that expects exactly 1 argument
iex> is_function(add, 1)
false
Closures
Anonymous functions can also access variables that are in scope when the function is defined. This is typically referred to as closures, as they close over their scope. Let's define a new anonymous function that uses the add
anonymous function we have previously defined:
iex> double = fn a -> add.(a, a) end
#Function<6.71889879/1 in :erl_eval.expr/5>
iex> double.(2)
4
A variable assigned inside a function does not affect its surrounding environment:
iex> x = 42
42
iex> (fn -> x = 0 end).()
0
iex> x
42
Clauses and guards
Similar to case/2
, we can pattern match on the arguments of anonymous functions as well as define multiple clauses and guards:
iex> f = fn
...> x, y when x > 0 -> x + y
...> x, y -> x * y
...> end
#Function<12.71889879/2 in :erl_eval.expr/5>
iex> f.(1, 3)
4
iex> f.(-1, 3)
-3
The number of arguments in each anonymous function clause needs to be the same, otherwise an error is raised.
iex> f2 = fn
...> x, y when x > 0 -> x + y
...> x, y, z -> x * y + z
...> end
** (CompileError) iex:1: cannot mix clauses with different arities in anonymous functions
The capture operator
Throughout this guide, we have been using the notation name/arity
to refer to functions. It happens that this notation can actually be used to capture an existing function into a data-type we can pass around, similar to how anonymous functions behave.
iex> fun = &is_atom/1
&:erlang.is_atom/1
iex> is_function(fun)
true
iex> fun.(:hello)
true
iex> fun.(123)
false
As you can see, once a function is captured, we can pass it as argument or invoke it using the anonymous function notation. The returned value above also hints we can capture functions defined in modules:
iex> fun = &String.length/1
&String.length/1
iex> fun.("hello")
5
You can also capture operators:
iex> add = &+/2
&:erlang.+/2
iex> add.(1, 2)
3
The capture syntax can also be used as a shortcut for creating functions. This is handy when you want to create functions that are mostly wrapping existing functions or operators:
iex> fun = &(&1 + 1)
#Function<6.71889879/1 in :erl_eval.expr/5>
iex> fun.(1)
2
iex> fun2 = &"Good #{&1}"
#Function<6.127694169/1 in :erl_eval.expr/5>
iex> fun2.("morning")
"Good morning"
The &1
represents the first argument passed into the function. &(&1 + 1)
above is exactly the same as fn x -> x + 1 end
. You can read more about the capture operator &
in its documentation.
Next let's revisit some of the data-types we learned in the past and dig deeper into how they work.