Quando precisamos tomar decisões difíceis e complicadas em nosso código, é sempre bom utilizar uma das estuturas de decisão.
No Elixir, estes foram criados para serem utilizados dentro de contextos com a devida semântica. Ou seja, pode-se dizer que existem estruturas de decisões que fazem a mesma coisa, só tem uma forma diferente de se entender por sua escrita.
O famoso if continua existindo no Elixir assim como em todas as outras, mas o que diferencia é que não é possível realizar vários else if
, ou seja, você na verdade precisa criar um bloco de if
dentro da definição do else
:
defmodule Decisao do
def esta_decidido?() do
if 1 == 2 do
true
else
if 1 == 3 do
true
else
if 1 == 1 do
true
else
false
end
end
end
end
end
A não ser que, você esteja atrás de uma semântica interessante, você deve utilizar essa estrutura.
Brincadeiras à parte, essa estrutura de decisão é exatamente o que o nome diz, uma validação ao contrário. Ou seja, caso a definição da condição tenha como resultado um booleano true
, o bloco de código definido será executado:
iex> hit_or_miss? = false
iex> unless hit_or_miss? === true do
...> IO.puts "Sim, eu sou o diferentão"
...> IO.puts "A não ser que você esteja errado ;)"
...> end
"Sim, eu sou o diferentão"
"A não ser que você esteja errado ;)"
Quando precisamos analisar inúmeras possíveis respostas de uma função, a estrutura de case
é a melhor por conta de ser a mais simples e ter uma semântica mais clean
para tal:
iex> case AqueleModulo.com_uma_resposta_muito_doida() do
...> "resposta 1" ->
...> "que loucura bicho"
...>
...> :atomo_muito_loco ->
...> "que loucura bicho"
...>
...> %{name: "Desestruct memo"} ->
...> "que loucura bicho"
...>
...> _qualquer_outra_resposta_mirabolante ->
...> "ai tu cagou tudo mermão"
...> end
Não é o Conde Drácula, mas esse aqui serve pra quando você precisa validar várias condições sem a necessidade de if .. else .. if .. else
, etc.
iex> cond do
...> 1 == 2 ->
...> "esse aqui é utopia, igual ancap"
...>
...> 1 == 3 ->
...> "esse aqui é utopia, igual ancap"
...>
...> 1 == 4 ->
...> "esse aqui é utopia, igual ancap"
...>
...> 1 == 0 ->
...> "esse aqui é utopia, igual ancap"
...>
...> true ->
...> "acho que não teve jeito mesmo"
...> end
Não é um with ela(e), mas é uma função do Kernel
para avaliar uma ou mais condições, é recomendado utilizar em momentos que seu fluxo de execução requerir validação de pattern matching de inúmeras condições onde o case
não seria o suficiente, principalmente quando você não precisa lidar com os resultados não esperados.
Dado uma função, se o seu retorno for o esperado (para isso será utilizado o pattern matching), execute o código. Ficará mais fácil entender com o exemplo a seguir:
iex> with {:ok, _banana} <- MyModule.my_fun(my_param) do
...> "Deu boa patrão"
...> end
Basicamente, se a função retornar uma tupla de {:ok, qualquer_valor}
(valores que você pode definir como quiser), o trecho de código no bloco do
será executado. Caso contrario, um erro é retornado para quem chamou a função.
O with
pode receber várias condições para avaliar, e também pode conter um bloco de else
para tratar qualquer cláusula que não seja esperada no bloco with ... do
. Para tratar tais cláusulas você pode usar uma espécie de case
.
iex> with {:ok, _banana} <- Modulo.funcao(),
...> _outra_banana <- Modulo.outra_funcao() do
...> "Deu boa patrão"
...> else
...> :error ->
...> "Deu merda patrão"
...>
...> _other_error ->
...> "Deu outra merda patrão"
...> end
Também é possível capturar os valores esperados utilizando o operador de atribuição.
iex> with %Struct{} = struct <- MyModule.my_fun(my_param) do
...> "Capturado chefia, #{struct}"
...> end
Um outro bom exemplo de uso do with ... do
é quando você não precisa tratar os valores não esperados que você teria de tratar com o case .. do
:
# Case
case Modulo.funcao() do
{:ok, _banana} -> "banana"
{:error, _motivo} -> "maçã"
_ -> "limão"
end
# With
with {:ok, _banana} <- Modulo.funcao() do
"Se der errado o problema já não é meu..."
end
Quando retornar {:error, _}
ou qualquer outro resultado, ele pode lidar isso em outro ponto do fluxo da execução, não necessitando de lidar com todos os retornos em todas as funções, mantendo tudo isso direto em apenas um único local.