🔥码云GVP开源项目 12k star Uniapp+ElementUI 功能强大 支持多语言、二开方便! 广告
14-模块属性 =========== [作为注释](#141-%E4%BD%9C%E4%B8%BA%E6%B3%A8%E9%87%8A) [作为常量](#142-%E4%BD%9C%E4%B8%BA%E5%B8%B8%E9%87%8F) [作为临时存储](#143-%E4%BD%9C%E4%B8%BA%E4%B8%B4%E6%97%B6%E5%AD%98%E5%82%A8) 在Elixir中,模块属性(attributes)主要服务于三个目的: 1. 作为一个模块的注释,通常附加上用户或虚拟机用到的信息 2. 作为常量 3. 在编译时作为一个临时的存储机制 让我们一个一个讲解。 ## 14.1-作为注释 Elixir从Erlang带来了模块属性的概念。例子: ```elixir 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文件中: ```elixir 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命令读取模块的注释: ```elixir $ 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工具](https://github.com/elixir-lang/ex_doc), 利用注释生成HTML页文档。 你可以看看[模块](http://elixir-lang.org/docs/stable/elixir/Module.html) 里面列出的模块属性列表,看看Elixir还支持那些模块属性。 Elixir还是用这些属性来定义 [typespecs](http://elixir-lang.org/docs/stable/elixir/Kernel.Typespec.html): - @spec 为一个函数提供specification - @callback 为行为回调函数提供spec - @type 定义一个@spec中用到的类型 - @typep 定义一个私有类型,用于@spec - @opaque 定义一个opaque类型用于@spec 本节讲了一些内置的属性。当然,属性可以被开发者、被一些类库扩展用来支持自定义的行为。 ## 14.2-作为常量 Elixir开发者经常会将模块属性当作常量定义使用: ```elixir defmodule MyServer do @initial_state %{host: "147.0.0.1", port: 3456} IO.inspect @initial_state end ``` >不同于Erlang,默认情况下用户定义的属性不会被存储在模块里。属性值仅在编译时存在。 开发者可以通过调用```Module.register_attribute/3```来使属性的行为更接近Erlang。 访问一个未定义的属性会报警告: ```elixir defmodule MyServer do @unknown end warning: undefined module attribute @unknown, please remove access to @unknown or explicitly set it to nil before access ``` 最后,属性也可以在函数中被读取: ```elixir 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](https://github.com/elixir-lang/plug)。 这个项目的目标是创建一个通用的Web库和框架。 >类似于ruby的rack Plug库允许开发者定义它们自己的plug,可以在一个web服务器上运行: ```elixir 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框架,它使用模块属性作为注释和存储: ```elixir defmodule MyTest do use ExUnit.Case @tag :external test "contacts external service" do # ... end end ``` ExUnit中,@tag标签被用来注释该测试用例。之后,这些标签可以作为过滤测试用例之用。 例如,你可以避免执行那些被标记成```:external```的测试,因为它们执行起来很慢。 本章带你一窥Elixir元编程的冰山一角,讲解了模块属性在开发中是如何扮演关键角色的。 下一章将讲解结构体和协议。