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

Foo*_*o42 12 macros elixir

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

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

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

Gaz*_*ler 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]}]]]}
Run Code Online (Sandbox Code Playgroud)

然后,您可以使用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"
Run Code Online (Sandbox Code Playgroud)

然后,您可以使用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
Run Code Online (Sandbox Code Playgroud)

  • 如果我的宏位于模块中,但我仍然想看看它扩展成这样,我该怎么做? (2认同)

Mar*_*jic 6

尝试Chris McCord的这个技巧:

your_ast |> Macro.expand(__ENV__) |> Macro.to_string |> IO.puts
Run Code Online (Sandbox Code Playgroud)