美文网首页
Elixir-集合

Elixir-集合

作者: 你期待的花开 | 来源:发表于2018-11-14 17:35 被阅读20次

列表、元组、关键字列表(keywords)、图(maps)、字典和函数组合子(combinators)

目录

  • 列表
    • 列表拼接
    • 列表减法
    • 头/尾
  • 元组
  • 关键字列表

列表

列表是值的简单集合,可以包括不同的数据类型,而且可能包含相同的值。

iex> [3.14, :pie, "Apple"]
[3.14, :pie, "Apple"]

Elixir 内部用链表实现列表,这表明获取列表长度是O(n)的操作。同样的原因,在头部插入比在尾部插入要快。

iex> list = [3.14, :pie, "Apple"]
[3.14, :pie, "Apple"]
iex> ["π"] ++ list
["π", 3.14, :pie, "Apple"]
iex> list ++ ["Cherry"]
[3.14, :pie, "Apple", "Cherry"]

列表的拼接

列表拼接使用 ++/2 操作符:

iex> [1, 2] ++ [3, 4, 1]
[1, 2, 3, 4, 1]

++/2的格式说明:在 Elixir 中(以及 Elixir 的基础语言 Erlang),函数和操作符的名字由两部分组成:名字(比如这里的 ++)和元数。元数是 Elixir 和 Erlang 代码非常核心的部分,它代表了给定行数接受的参数个数(比如这里的 2),元数和名字之间通过斜线分割。

列表减法

--/2 操作符支持列表的减法,而且减去不存在的值也是安全的。

iex> ["foo", :bar, 42] -- [42, "bar"]
["foo", :bar]

要注意重复值的处理:对于左边列表中 的每个值,右边只有首次出现的这个值会被删除:

iex> [1,2,2,3,2,3] -- [1,2,3,2]
[2, 3]

头/尾

使用列表的时候,经常要和头部和尾部打交道:列表的头部是列表的第一个元素;尾部是除去第一个元素剩下的列表。 Elixir 提供了两个函数 hdtl 来获取这两个部分。

iex> hd [3.14, :pie, "Apple"]
3.14
iex> tl [3.14, :pie, "Apple"]
[:pie, "Apple"]

除了上面的提到的函数,你还可以使用模式匹配(pattern matching)| 操作符来把一个列表分成头尾两部分;

iex> [head | tail] = [3.14, :pie, "Apple"]
[3.14, :pie, "Apple"]
iex> head
3.14
iex> tail
[:pie, "Apple"]

元组

元组和列表很像,但元组在内存是连续存放的。这样,获取元组的长度很快,但修改元组的操作很昂贵,新的元组必须重新在内存中拷贝一份。定义元组要用花括号:

iex> {3.14, :pie, "Apple"}
{3.14, :pie, "Apple"}

元组一个很常见的用法是作为函数的返回值,来返回额外的信息,当介绍到模式匹配的时候,这种用法的好处就显而易见了。

iex> File.read("path/to/existing/file")
{:ok, "... contents ..."}
iex> File.read("path/to/unknown/file")
{:error, :enoent}

关键字列表

关键字列表(keywords)和图(maps)是Elixir 中两个相关的集合,Elixir 的关键字列表是一种独特的列表:列表里的内容是二元元组,并且二元组的第一个元素必须是原子。它和列表的行为完全一致。

iex> [foo: "bar", hello: "world"]
[foo: "bar", hello: "world"]
iex> [{:foo, "bar"}, {:hello, "world"}]
[foo: "bar", hello: "world"]

关键字列表非常重要,它有以下特性:

  • 键都是原子
  • 键是有序的(定义后,顺序不会改变)
  • 键不是唯一的

因为这些特性,关键字列表最常见的用法是作为参数传递给函数。

Elixir 的图示键值对结构的第一选择,和关键字列表不通,图允许任意类型的数据作为键,而且数据并不严格排序。你可以使用%{}来定义图:

iex> map = %{:foo => "bar", "hello" => :world}
%{:foo => "bar", "hello" => :world}
iex> map[:foo]
"bar"
iex> map["hello"]
:world

Elixir 1.2 版本中,也可以把变量作为图的键:

iex> key = "hello"
"hello"
iex> %{key => "world"}
%{"hello" => "world"}

如果重复的键添加到图中,后面的值会覆盖之前的值:

iex> %{:foo => "bar", :foo => "hello world"}
%{foo: "hello world"}

从上面的运行结果可以看出,存储键只有原子的图,可以不用 =>,直接使用关键字列表的语法:

iex> %{foo: "bar", hello: "world"}
%{foo: "bar", hello: "world"}

iex> %{foo: "bar", hello: "world"} == %{:foo => "bar", :hello => "world"}
true

图的另一个特性是:它们提供令人自己更新和获取原子键的语法:

iex> map = %{foo: "bar", hello: "world"}
%{foo: "bar", hello: "world"}
iex> %{map | foo: "baz"}
%{foo: "baz", hello: "world"}
iex> map.hello
"world"
iex> map[hello:]
"world"

相关文章

  • Elixir-集合

    列表、元组、关键字列表(keywords)、图(maps)、字典和函数组合子(combinators) 目录 列表...

  • Elixir-组合

    模块 模块是把函数组织到不同命名空间的最好的办法,除了能为函数分组,他还允许我们定义命名函数和私有函数。 Elix...

  • Elixir-函数

    目录 匿名函数& 操作符 模式匹配 命名函数函数名字和元数私有函数卫兵参数默认值 匿名函数 匿名函数就是没有名字,...

  • Elixir-推导

    在 Elixir 中,列表推导是循环遍历枚举值的语法糖。 基础 推导经常用来根据 Enum 和 Stream生...

  • Elixir-并发

    Elixir 的一大卖点就是对并发的支持,得益于 Erlang VM (BEAM) ,Elixir 的并发要比预期...

  • Elixir-基础

    安装 在https://elixir-lang.org 上可以找到安装说明。 使用 elixir -v 查看 el...

  • Elixir-魔符

    Elixir 提供了一种叫做 魔符的语法糖来标识和处理字面量。一个魔符已~开头然后接上一个字符。Elixir 已经...

  • Elixir-模式匹配

    模式匹配是 Elixir 很强大的特性,它允许我们匹配简单值、数据结构、甚至函数。 匹配操作符 Elixir 中,...

  • Elixir-控制语句

    if 和 unless 你之前可能遇到过 if/2了,如果你使用过 Ruby,也会很熟悉 unless。它们在 E...

  • Elixir-文档模块

    Elixir提供了多种方式来编写注释或者是注解。下面是其中三种方式: # - 用于单行的注释 @moduledoc...

网友评论

      本文标题:Elixir-集合

      本文链接:https://www.haomeiwen.com/subject/eqmtmftx.html