如何通过指定模块和方法名称在Elixir中动态调用方法?

时间:2012-11-05 11:15:45

标签: erlang metaprogramming elixir

我想知道elixir中的方法名称究竟是什么:

array = [1,2,3]
module_name = :lists
method_name = :nth                  # this not working
module_name.method_name(1, array)   # error, undef function lists.method_name/2
module_name.nth(1, array)           # returns 1, module_name is OK. It's an atom

但我可以在erlang中做同样的事情:

A = [1,2,3].
X = lists.
Y = nth.
X:Y(1,A).  #  returns 1

我怎样才能在灵药中做到这一点?

1 个答案:

答案 0 :(得分:45)

您可以使用apply/3,它只是:erlang.apply/3的包装。它只是invokes the given function from the module with an array of arguments.因为您将参数作为模块和函数名称传递,所以您可以使用变量。

apply(:lists, :nth, [1, [1,2,3]])
apply(module_name, method_name, [1, array])

如果您想了解有关elixir如何处理函数调用(以及其他所有内容)的更多信息,您应该查看quoteunquote

contents = quote do: unquote(module_name).unquote(method_name)(1, unquote(array))

返回函数调用的homoiconic表示。

{{:.,0,[:lists,:nth]},0,[1,[1,2,3]]}

您可以使用Code.eval_quoted/3

unquote引用的函数调用
{value, binding} = Code.eval_quoted(contents)

编辑:这是一个使用Enum.fetch和var。

的例子
quoted_fetch = quote do: Enum.fetch([1,2,3], var!(item));             
{value, binding} = Code.eval_quoted(quoted_fetch, [item: 2])