文档章节

elixir官方教程 元编程(一) 引用与去引用

ljzn
 ljzn
发布于 2016/08/13 21:10
字数 1352
阅读 56
收藏 0

#引用与去引用

  1. 引用(Quoting)
  2. 去引用(Unquoting)
  3. 释放(Escaping)

一个Elixir程序可以用它自己的数据结构来表现.本章,我们将会学习这些结构体的特点和如何组成它们.本章我们要学习的概念是为宏的积木(building blocks),在下一章中我们将深入研究它.

#引用(Quoting)

Elixir程序中的积木是一个三元素元组.例如,函数sum(1, 2, 3)的内部表示是:

{:sum, [], [1, 2, 3]}

你可以用quote宏来得到任何表达式的内部表现:

iex> quote do: sum(1, 2, 3)
{:sum, [], [1, 2, 3]}

第一个元素是函数名,第二个元素是一个包含了元数据的关键词列表,第三个元素是参数列表.

操作符也可以用元组来表示:

iex> quote do: 1 + 2
{:+, [context: Elixir, import: Kernel], [1, 2]}

甚至一个映射都被表示成对%{}的调用:

iex> quote do: %{1 => 2}
{:%{}, [], [{1, 2}]}

变量也能这样三段式表示,只不过最后的元素换成了一个原子:

iex> quote do: x
{:x, [], Elixir}

当引用更复杂的表达式时,代码被表示成了一个树状的嵌套元组.许多语言将这种表示称为抽象语法树(AST).ELixir称其为引用表达式:

iex> quote do: sum(1, 2 + 3, 4)
{:sum, [], [1, {:+, [context: Elixir, import: Kernel], [2, 3]}, 4]}

有时,把引用表达式还原为文本代码会很有用.可以用Macro.to_string/1来完成:

iex> Macro.to_string(quote do: sum(1, 2 + 3, 4))
"sum(1, 2 + 3, 4)"

通常,上述元组的结构会是这种格式:

{atom | tuple, list, list | atom}

- 第一个元素是一个原子,或者是同样表达方式的另一个元组; - 第二个元素是一个关键词列表,包含元数据,比如数字和语境; - 第三个元素是函数调用的参数列表或者是一个原子.当为原子时,意味着元组表示的是一个变量.

除了上面定义的元组,有五个Elixir字面量,在被引用时,会返回它们自己(而不是一个元组).它们是:

:sum         #=> Atoms
1.0          #=> Numbers
[1, 2]       #=> Lists
"strings"    #=> Strings
{key, value} #=> Tuples with two elements

大多数Elixir代码都有一个直截了当的引用表达式.我们建议你尝试不同的代码,看看结果如何.比如,String.upcase("foo")会如何展开?我们已经知道if(true, do: :this, else: :that)等同于if true do :this else :that end.它要如何用引用表达式来容纳?

#去引用(Unquoting)

引用可以获得一些特定代码块的内部表达式.然而,有时我们需要注入一些特定代码块到我们想要获取的表达式中.

例如,假设我们有一个变量,包含了我们想注入一个引用表达式中的数字,变量名为number.

iex> number = 13
iex> Macro.to_string(quote do: 11 + number)
"11 + number"

那不是我们想要的,number变量被引入了表达式,但number变量的值没有被注入.为了注入number变量的值,我们需要在引用表达式中使用unquote:

iex> number = 13
iex> Macro.to_string(quote do: 11 + unquote(number))
"11 + 13"

unquote甚至可以被用于注入函数名:

iex> fun = :hello
iex> Macro.to_string(quote do: unquote(fun)(:world))
"hello(:world)"

在相同的情况下,可能需要注入一个有许多值的列表.比如,假设你有一个列表[1, 2, 6],我们想把[3, 4, 5]注入进去.使用unquote却不能得到想要的结果:

iex> inner = [3, 4, 5]
iex> Macro.to_string(quote do: [1, 2, unquote(inner), 6])
"[1, 2, [3, 4, 5], 6]"

这时就轮到unquote_splicing出场了:

iex> inner = [3, 4, 5]
iex> Macro.to_string(quote do: [1, 2, unquote_splicing(inner), 6])
"[1, 2, 3, 4, 5, 6]"

去引用在操作宏的时候很有用.编写宏的时候,开发者可以获取代码块并将它们注入到其它代码块中,这可以被用于改变代码或者编写能在编译时生成代码的代码.

#释放(Escaping)

如本章开头所见,Elixir中只有一部分值有合法的引用表达式.比如,一个映射没有合法的引用表达式.四元素的元组也没有.然而,这些值可以被表示成一个引用表达式:

iex> quote do: %{1 => 2}
{:%{}, [], [{1, 2}]}

有时你需要将这种值注入引用表达式中.我们首先需要将这些值释放到引用表达式中,使用Macro.escape/1来完成:

iex> map = %{hello: :world}
iex> Macro.escape(map)
{:%{}, [], [hello: :world]}

宏接收引用表达式,并必须返回引用表达式.然而执行宏的过程中,你可能需要处理一些值,并且区分值与引用表达式.

也就是说,区分常规的Elixir值(例如列表,映射,进程,参考等等)与引用表达式,是很重要的.有一些值的引用表达式就是它们自己,例如整数,原子核字符串.另一些值,比如映射,需要被显式转换.最后,函数与参考完全不能被转化成引用表达式.

你可以在Kernel.SpecialForms模块中阅读更多关于quoteunquote的内容.在Macro模块中可以找到Macro.escape/1的文档和其它与引用表达式相关的函数.

在本教程中我们将编写自己的第一个宏,让我们进入下一章吧.

© 著作权归作者所有

共有 人打赏支持
ljzn
粉丝 29
博文 69
码字总数 96245
作品 0
南平
程序员
私信 提问
elixir官方教程 元编程(二) 宏

宏 前言 我们的第一个宏 宏的隔离 环境 私有宏 负责任地编写宏 前言 尽管Elixir已竭力为宏提供一个安全的环境,用宏编写干净代码的责任仍然落在了开发者身上.宏比传统的Elixir函数更难编写,而...

ljzn
2016/08/14
1K
3
elixir官方入门教程 学习资料

下一步该去哪 构建你的第一个Elixir项目 元编程 社区与其它资源 Erlang基础 想要学习更多?继续阅读! 构建你的第一个Elixir项目 为了开始你的第一个项目,Elixir装载了一个叫做Mix的构建工具....

ljzn
2016/08/06
144
0
Elixir学习笔记(模型匹配、控制语句)

模型匹配 模式匹配是 Elixir 很强大的特性,它允许我们匹配简单值、数据结构、甚至函数。 匹配运算符 在Elixir中,运算符实际上叫做 匹配运算符。通过这个匹配操作符,我们可以赋值和匹配值。...

程序员小哥哥
04/12
0
0
Elixir 学习资源

Elixir 官网 getting started官方入门学习资源 官方文档 hex 包管理系统 elixir sips 比较不错视频课程 Elixir China 中文论坛 官方wiki Elixir by Example Awesome Elixir Elixir Quiz 通过...

lidashuang
2017/11/29
0
0
Elixir: 编程语言的未来

这篇文章谈一谈最近火爆的 Elixir,同时说一下对编程语言选择的看法。同时作为 Erlang 发烧友,Elixir 不可不提。即使有了那么多编程语言 Elixir 也值得接触。 现在开始接触 Elixir 对编程语...

oschina
2015/08/02
6.6K
18

没有更多内容

加载失败,请刷新页面

加载更多

关于ElasticSearch的使用过程遇到的问题

由于作者从官网下载了ES5.6.10的安装包,解压之后就开始运行ES,前面一切正常。 后面某个查询条件失效。 解决: 1.先试了把单个查询条件撤离出来,当成一个Test来跑,发现还是获取不到值,表...

DoLo-lty
4分钟前
0
0
sed 替换文本内得路径字符等等

1. 句子 sed -i 's%/opt/apache-maven-3.5.3/conf/settings.xml%/data/opt/apache-maven-3.5.3/conf/settings.xml%g' ./*/config.xml 2. 解释 sed linux 一个文件流式处理的工具 2.1 -i 在当......

Aruforce
5分钟前
0
0
mysql_索引

索引类型 哈希表 有序数组 搜索树 MySQL索引 B-树 B+树 innodb的索引 索引维护 关于自增主键的使用 参考 极客时间《mysql实战45讲》

grace_233
5分钟前
0
0
“入乡随俗,服务为主” 发明者量化兼容麦语言啦!

5年时光 我们裹挟前行。发明者量化从筚路蓝缕到步履蹒跚,从以“区块链资产交易”为阵地,再到以“内外盘商品期货”为依托。再到今天全面兼容“麦语言”。每一步,我们始终都在为建立一个优秀...

酒逢知己千杯少
7分钟前
0
0
session深入探讨

简介 session,会话,其实是一个容易让人误解的词。它总跟web系统的会话挂钩,利用session,javaweb项目实现了登录状态的控制。坊间流传,关闭浏览器,就是关闭了web系统的会话。其实浏览器对...

千里明月
8分钟前
0
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部