和Erlang互操作

在 Erlang VM (BEAM) 上构建 Elixir 的好处之一就是已经有大量的库可以供我们使用。互操作性允许我们在 Elixir 代码中直接使用 Erlang 的标准库和三方库。这节课,我们就讲讲如何来做。

标准库

在任何 Elixir 代码中都能直接使用 Erlang 提供的标准库,Erlang 的模块用小写的原子变量表示,比如 :os:timer

我们可以用 timer.tc 计算某个函数执行的时间:

defmodule Example do
  def timed(fun, args) do
    {time, result} = :timer.tc(fun, args)
    IO.puts("Time: #{time} μs")
    IO.puts("Result: #{result}")
  end
end

iex> Example.timed(fn (n) -> (n * n) * n end, [100])
Time: 8 μs
Result: 1000000

要了解所有可用的模块,请看 Erlang 参考手册

Erlang 第三方依赖

在之前的课程中,我们讲过如何使用 Mix 和管理依赖。要引入 Erlang 的依赖,方法也是一样的。如果依赖的 Erlang 库不在 hex,你也可以直接使用 git 代码库的地址:

def deps do
  [{:png, github: "yuce/png"}]
end

然后我们就可以用 Erlang 的库了:

png =
  :png.create(%{:size => {30, 30}, :mode => {:indexed, 8}, :file => file, :palette => palette})

区别

知道了怎么在 Elixir 中使用 Erlang ,我们还要讲讲操作 Erlang 语言会遇到的坑:

原子

Erlang 的原子和 Elixir 很相似,只是没有前面的冒号(:),Erlang 中的原子是小写字母和下划线的组合。

Elixir:

:example

Erlang:

example.

字符串

在 Elixir 里面,字符串表达的是 UTF-8 编码的二进制数据。而在 Erlang,字符串还是使用双引号表示,但是却是字符列表。

Elixir:

iex> is_list('Example')
true
iex> is_list("Example")
false
iex> is_binary("Example")
true
iex> <<"Example">> === "Example"
true

Erlang:

1> is_list('Example').
false
2> is_list("Example").
true
3> is_binary("Example").
false
4> is_binary(<<"Example">>).
true

需要特别注意的是,有些 Erlang 的库不支持二进制数据,我们要把 Elixir 字符串转换成字符列表。还好,to_charlist/1 函数可以帮我们轻松完成这个转换。

iex> :string.words("Hello World")
** (FunctionClauseError) no function clause matching in :string.strip_left/2

    The following arguments were given to :string.strip_left/2:

        # 1
        "Hello World"

        # 2
        32

    (stdlib) string.erl:1661: :string.strip_left/2
    (stdlib) string.erl:1659: :string.strip/3
    (stdlib) string.erl:1597: :string.words/2

iex> "Hello World" |> to_charlist() |> :string.words()
2

变量

Elixir:

iex> x = 10
10

iex> x1 = x + 10
20

Erlang:

1> X = 10.
10

2> X1 = X + 1.
11

就这么多!在 Elixir 应用中使用 Erlang 简单高效,把原来可用的库直接加倍啦!

最后更新于