Эликсир - Перечислимые
Перечислимый объект - это объект, который можно перечислить. «Перечислимый» означает отсчет членов набора / коллекции / категории один за другим (обычно по порядку, обычно по имени).
Elixir предоставляет концепцию перечисляемых объектов и модуль Enum для работы с ними. Функции в модуле Enum ограничены, как следует из названия, перечислением значений в структурах данных. Примером перечисляемой структуры данных является список, кортеж, карта и т. Д. Модуль Enum предоставляет нам чуть более 100 функций для работы с перечислениями. В этой главе мы обсудим несколько важных функций.
Все эти функции принимают перечислимое число в качестве первого элемента и функцию в качестве второго и работают с ними. Функции описаны ниже.
все?
Когда мы используем all? функция, вся коллекция должна быть истинной, иначе будет возвращено ложь. Например, чтобы проверить, все ли элементы в списке являются нечетными числами, тогда.
res = Enum.all?([1, 2, 3, 4], fn(s) -> rem(s,2) == 1 end)
IO.puts(res)
Когда вышеуказанная программа запускается, она дает следующий результат -
false
Это потому, что не все элементы в этом списке нечетные.
Любые?
Как следует из названия, эта функция возвращает true, если какой-либо элемент коллекции имеет значение true. Например -
res = Enum.any?([1, 2, 3, 4], fn(s) -> rem(s,2) == 1 end)
IO.puts(res)
Когда вышеуказанная программа запускается, она дает следующий результат -
true
кусок
Эта функция делит нашу коллекцию на небольшие части, размер которых указан во втором аргументе. Например -
res = Enum.chunk([1, 2, 3, 4, 5, 6], 2)
IO.puts(res)
Когда вышеуказанная программа запускается, она дает следующий результат -
[[1, 2], [3, 4], [5, 6]]
каждый
Может возникнуть необходимость перебрать коллекцию без создания нового значения, в этом случае мы используем each функция -
Enum.each(["Hello", "Every", "one"], fn(s) -> IO.puts(s) end)
Когда вышеуказанная программа запускается, она дает следующий результат -
Hello
Every
one
карта
Чтобы применить нашу функцию к каждому элементу и создать новую коллекцию, мы используем функцию карты. Это одна из самых полезных конструкций в функциональном программировании, поскольку она достаточно выразительна и коротка. Давайте рассмотрим пример, чтобы понять это. Мы удвоим значения, хранящиеся в списке, и сохраним их в новом спискеres -
res = Enum.map([2, 5, 3, 6], fn(a) -> a*2 end)
IO.puts(res)
Когда вышеуказанная программа запускается, она дает следующий результат -
[4, 10, 6, 12]
уменьшить
В reduceФункция помогает нам сократить количество перечисляемых до единственного значения. Для этого мы предоставляем дополнительный аккумулятор (5 в этом примере), который будет передан в нашу функцию; если аккумулятор не указан, используется первое значение -
res = Enum.reduce([1, 2, 3, 4], 5, fn(x, accum) -> x + accum end)
IO.puts(res)
Когда вышеуказанная программа запускается, она дает следующий результат -
15
Аккумулятор - это начальное значение, передаваемое в fn. Начиная со второго вызова значение, возвращенное из предыдущего вызова, передается как аккумулятор. Мы также можем использовать сокращение без аккумулятора -
res = Enum.reduce([1, 2, 3, 4], fn(x, accum) -> x + accum end)
IO.puts(res)
Когда вышеуказанная программа запускается, она дает следующий результат -
10
уникальный
Функция uniq удаляет дубликаты из нашей коллекции и возвращает только набор элементов в коллекции. Например -
res = Enum.uniq([1, 2, 2, 3, 3, 3, 4, 4, 4, 4])
IO.puts(res)
При запуске вышеуказанной программы он дает следующий результат -
[1, 2, 3, 4]
Жадная оценка
Все функции в модуле Enum готовы. Многие функции ожидают перечислимого и возвращают список. Это означает, что при выполнении нескольких операций с Enum каждая операция будет генерировать промежуточный список, пока мы не достигнем результата. Давайте рассмотрим следующий пример, чтобы понять это -
odd? = &(odd? = &(rem(&1, 2) != 0)
res = 1..100_000 |> Enum.map(&(&1 * 3)) |> Enum.filter(odd?) |> Enum.sum
IO.puts(res)
Когда вышеуказанная программа запускается, она дает следующий результат -
7500000000
В приведенном выше примере есть конвейер операций. Мы начинаем с диапазона, а затем умножаем каждый элемент в диапазоне на 3. Эта первая операция теперь создаст и вернет список из 100_000 элементов. Затем мы сохраняем все нечетные элементы из списка, генерируя новый список, теперь с 50_000 элементами, а затем суммируем все записи.
В |> символ, использованный в приведенном выше фрагменте, - это pipe operator: он просто берет результат выражения в левой части и передает его в качестве первого аргумента в вызов функции в правой части. Это похоже на Unix | оператор. Его цель - выделить поток данных, преобразовываемых рядом функций.
Без pipe оператор, код выглядит сложным -
Enum.sum(Enum.filter(Enum.map(1..100_000, &(&1 * 3)), odd?))
У нас есть много других функций, однако здесь описано лишь несколько важных.