Elixir Operator重载:使用宏定义独占范围

问题描述 投票:2回答:1

为了更好地理解Elixir语言,我想尝试通过添加一个独特的范围运算符...来进行运算符重载。一个例子:1...10然后会创建一个从1到10但不包括10的范围。 (例如1...10 == 1..9

所以,我查了the definition of ..,因为...的功能当然非常相似。

Mmy模块然后变成:

defmodule Sequences.Ranges do

  defmacro first ... last do
    case is_float(first) or is_float(last) or
         is_atom(first) or is_atom(last) or
         is_binary(first) or is_binary(last) or
         is_list(first) or is_list(last) do
      true ->
        raise ArgumentError,
          "ranges (first...last) expect both sides to be integers, " <>
          "got: #{Macro.to_string({:"Sequences.Ranges...", [], [first, last]})}"
      false ->
        case __CALLER__.context do
          nil -> quote do: Elixir.Range.new(unquote(first), unquote(last-1))
          _   -> {:%{}, [], [__struct__: Elixir.Range, first: first, last: (last-1)]}
        end
    end
  end

end

但是,在编译此模块时,我收到以下错误:

== Compilation error on file lib/sequences/ranges.ex ==
** (CompileError) lib/sequences/ranges.ex:6: cannot invoke local .../1 inside match
    (stdlib) lists.erl:1353: :lists.mapfoldl/3

我究竟做错了什么?

macros operator-overloading elixir
1个回答
5
投票

在Elixir中只能定义有限数量的中缀运算符:

\\, <-, |, ~>>, <<~, ~>, <~, <~>, <|>, <<<, >>>, |||, &&&, and ^^^

它们在解析器中预定义,但未实现(或者至少默认情况下不导入)。这使它们可用于自定义实现,但除非您修改并重新编译Elixir本身,否则无法创建自己的添加到此列表。

...不是其中之一,所以它不会起作用。

来自评论:

|||&&&^^^<<<>>>~~~默认情况下不会导入,但它们是默认Elixir标准库的Bitwise模块的一部分。

有关详细信息,请参阅http://www.rodneyfolz.com/custom-infix-functions-in-elixir/

© www.soinside.com 2019 - 2024. All rights reserved.