19

在 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
4

2 回答 2

25

简短的回答是:你不能。

长答案是您可以定义宏,只要它们由有效的守卫表达式组成,它们就可以在守卫中工作。但这对于复杂的守卫来说并非易事,所以除非你真的必须这样做,否则最好避免。

可以在入门指南中找到有效守卫的列表: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

于 2014-01-21T19:31:45.053 回答
2

您现在可以使用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

于 2018-07-02T06:06:14.410 回答