在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
答案 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