创建新的保护条款

时间:2014-01-21 15:02:03

标签: function pattern-matching elixir

在Elixir中,我将如何为函数创建新的保护条款?显然,我已经看到你不能只在when语句中调用任何函数,但能够做到这样的事情会很好:

defmodule Player do
  def play_card(player), do: []
  def play_card(player) when has_cards(player), do: ...
  # Define has_cards as guard clause?
end

2 个答案:

答案 0 :(得分:23)

简短的回答是:你做不到。

答案很长,你可以定义宏,只要它们是由有效的防护表达式构成的,它们就可以在防护中工作。但对于复杂的守卫来说,这是非常重要的,所以除非你真的需要,否则最好避免使用。

有效警卫列表可在入门指南中找到:http://elixir-lang.org/getting-started/case-cond-and-if.html#expressions-in-guard-clauses

可以在Elixir源中找到“复杂”保护宏的示例实现:https://github.com/elixir-lang/elixir/blob/5d34fcdccd20bf38c4f2a13f8adeb78e22b4e74c/lib/elixir/lib/kernel.ex#L1574

答案 1 :(得分:1)

您可以使用Kernel.defguard/1(或defguardp/1)立即执行此操作:

defmodule Player do
  defstruct [:name, cards: []]
  defguard has_cards(cards) when cards != []

  def play_card(%__MODULE__{cards: cards}) when has_cards(cards), do: []
  def play_card(_player), do: []
end

文档:https://hexdocs.pm/elixir/Kernel.html#defguard/1

但是,必要性防护仍然是非常简单的构造,因此您不能真正将复杂的逻辑放入其中。作为参考,可以使用以下防护措施:https://hexdocs.pm/elixir/master/guards.html#list-of-allowed-expressions