有没有简单的方法来查看Elixir宏扩展到什么?

时间:2016-02-04 15:22:06

标签: macros elixir

在过去18个月左右的时间里,Elixir一直是我的goto语言,但我有时会发现“没有魔法”的口头禅(特别是引用凤凰与Rails的引用)和宏的使用之间存在紧张关系。

虽然当我使用没有它们的语言时我现在想念宏,但我仍然希望看到它们实际上在做什么更容易。我的某些部分总是希望拉回DSL幕后看到真实的代码。

是否有一种简单的方法可以扩展宏并查看它们生成的代码(也许是通过IEx),这样我就不必深入研究defmacro层,试图将它拼凑在一起。

2 个答案:

答案 0 :(得分:11)

您可以使用Macro.expand/2

展开宏
iex> Macro.expand((quote do: (if true, do: 1)), __ENV__)
{:case, [optimize_boolean: true],
 [true,
  [do: [{:->, [],
     [[{:when, [],
        [{:x, [counter: 6], Kernel},
         {:in, [context: Kernel, import: Kernel],
          [{:x, [counter: 6], Kernel}, [false, nil]]}]}], nil]},
    {:->, [], [[{:_, [], Kernel}], 1]}]]]}

然后,您可以使用Macro.to_string/2将输出作为字符串而不是AST:

iex> Macro.expand((quote do: (if true, do: 1)), __ENV__) |> Macro.to_string()
"case(true) do\n  x when x in [false, nil] ->\n    nil\n  _ ->\n    1\nend"

然后,您可以使用IO.puts/2将字符串打印到终端:

iex> Macro.expand((quote do: (if true, do: 1)), __ENV__) |> Macro.to_string() |> IO.puts()
case(true) do
  x when x in [false, nil] ->
    nil
  _ ->
    1
end

答案 1 :(得分:6)

尝试克里斯麦考德的这个技巧:

your_ast |> Macro.expand(__ENV__) |> Macro.to_string |> IO.puts