14-模块属性

作为注释
作为常量
作为临时存储

在Elixir中,模块属性(attributes)主要服务于三个目的:

  1. 作为一个模块的注释,通常附加上用户或虚拟机用到的信息
  2. 作为常量
  3. 在编译时作为一个临时的存储机制

让我们一个一个讲解。

14.1-作为注释

Elixir从Erlang带来了模块属性的概念。例子:

defmodule MyServer do
  @vsn 2
end

这个例子中,我们显式地为该模块设置了 版本(vsn即version) 属性。 属性标识@vsn是预定义的属性名称,会被Erlang虚拟机的代码装载机制使用: 读取并检查该模块是否在某处被更新了。 如果不注明版本号,会被自动设置为这个模块函数的md5 checksum。

Elixir有个好多系统保留的预定义属性。比如一些常用的:

  • @moduledoc 为整个模块提供文档说明
  • @doc 为该属性后面的函数或宏提供文档说明
  • @behaviour (注意这个单词是英式拼法)用来注明一个OTP或用户自定义行为
  • @before_compile 提供一个每当模块被编译之前执行的钩子。这使得我们可以在模块被编译之前往里面注入函数。

@moduledoc和@doc是很常用的属性,推荐经常使用(写文档)。

Elixir视文档为一等公民,提供了很多方法来访问文档。

让我们回到上几章定义的Math模块,为它添加文档,然后依然保存在math.ex文件中:

defmodule Math do
  @moduledoc """
  Provides math-related functions.

  ## Examples

      iex> Math.sum(1, 2)
      3

  """

  @doc """
  Calculates the sum of two numbers.
  """
  def sum(a, b), do: a + b
end

上面例子使用了heredocs注释。heredocs是多行的文本,用三个引号包裹,保持里面内容的格式。 下面例子演示在iex中,用h命令读取模块的注释:

$ elixirc math.ex
$ iex
iex> h Math # Access the docs for the module Math
...
iex> h Math.sum # Access the docs for the sum function
...

Elixir还提供了ExDoc工具, 利用注释生成HTML页文档。

你可以看看模块 里面列出的模块属性列表,看看Elixir还支持那些模块属性。

Elixir还是用这些属性来定义 typespecs

  • @spec 为一个函数提供specification
  • @callback 为行为回调函数提供spec
  • @type 定义一个@spec中用到的类型
  • @typep 定义一个私有类型,用于@spec
  • @opaque 定义一个opaque类型用于@spec

本节讲了一些内置的属性。当然,属性可以被开发者、被一些类库扩展用来支持自定义的行为。

14.2-作为常量

Elixir开发者经常会将模块属性当作常量定义使用:

defmodule MyServer do
  @initial_state %{host: "147.0.0.1", port: 3456}
  IO.inspect @initial_state
end

不同于Erlang,默认情况下用户定义的属性不会被存储在模块里。属性值仅在编译时存在。 开发者可以通过调用Module.register_attribute/3来使属性的行为更接近Erlang。

访问一个未定义的属性会报警告:

defmodule MyServer do
  @unknown
end
warning: undefined module attribute @unknown, please remove access to @unknown or explicitly set it to nil before access

最后,属性也可以在函数中被读取:

defmodule MyServer do
  @my_data 14
  def first_data, do: @my_data
  @my_data 13
  def second_data, do: @my_data
end

MyServer.first_data #=> 14
MyServer.second_data #=> 13

注意,在函数内读取某属性,读取的是该属性当前值的快照。换句话说,读取的是编译时的值,而非运行时。 后面我们将看到,这个特点使得属性可以作为模块在编译时的临时存储。

14.3-作为临时存储

Elixir组织中有一个项目,叫做Plug。 这个项目的目标是创建一个通用的Web库和框架。

类似于ruby的rack

Plug库允许开发者定义它们自己的plug,可以在一个web服务器上运行:

defmodule MyPlug do
  use Plug.Builder

  plug :set_header
  plug :send_ok

  def set_header(conn, _opts) do
    put_resp_header(conn, "x-header", "set")
  end

  def send_ok(conn, _opts) do
    send(conn, 200, "ok")
  end
end

IO.puts "Running MyPlug with Cowboy on http://localhost:4000"
Plug.Adapters.Cowboy.http MyPlug, []

上面例子我们用了plug/1宏来连接各个在处理请求时会被调用的函数。 在内部,每当你调用plug/1时,Plug把参数存储在@plug属性里。 在模块被编译之前,Plug执行一个回调函数,这个函数定义了处理http请求的方法。 这个方法将顺序执行所有保存在@plug属性里的plugs。

为了理解底层的代码,我们需要宏。因此我们将回顾一下元编程手册里这种模式。 但是这里的重点是怎样使用属性来存储数据,让开发者得以创建DSL(领域特定语言)。

另一个例子来自ExUnit框架,它使用模块属性作为注释和存储:

defmodule MyTest do
  use ExUnit.Case

  @tag :external
  test "contacts external service" do
    # ...
  end
end

ExUnit中,@tag标签被用来注释该测试用例。之后,这些标签可以作为过滤测试用例之用。 例如,你可以避免执行那些被标记成:external的测试,因为它们执行起来很慢。

本章带你一窥Elixir元编程的冰山一角,讲解了模块属性在开发中是如何扮演关键角色的。
下一章将讲解结构体和协议。