Эликсир - Протоколы

Протоколы - это механизм для достижения полиморфизма в Elixir. Диспетчеризация по протоколу доступна для любого типа данных, если он реализует протокол.

Рассмотрим пример использования протоколов. Мы использовали функцию под названиемto_stringв предыдущих главах для преобразования из других типов в строковый тип. На самом деле это протокол. Он действует в соответствии с введенными данными, не вызывая ошибок. Может показаться, что мы обсуждаем функции сопоставления с образцом, но по мере того, как мы продолжим, выясняется, что все по-другому.

Рассмотрим следующий пример, чтобы лучше понять механизм протокола.

Давайте создадим протокол, который будет отображать, является ли данный вход пустым или нет. Мы будем называть этот протоколblank?.

Определение протокола

Мы можем определить протокол в Elixir следующим образом:

defprotocol Blank do
   def blank?(data)
end

Как видите, нам не нужно определять тело функции. Если вы знакомы с интерфейсами на других языках программирования, вы можете думать о протоколе как о том же самом.

Итак, этот Протокол говорит, что все, что его реализует, должно иметь empty?функция, хотя разработчик должен решать, как функция отреагирует. Определив протокол, давайте разберемся, как добавить пару реализаций.

Реализация протокола

Поскольку мы определили протокол, теперь нам нужно указать ему, как обрабатывать различные входные данные, которые он может получать. Возьмем за основу взятый ранее пример. Мы реализуем пустой протокол для списков, карт и строк. Это покажет, является ли то, что мы передали, пустым или нет.

#Defining the protocol
defprotocol Blank do
   def blank?(data)
end

#Implementing the protocol for lists
defimpl Blank, for: List do
   def blank?([]), do: true
   def blank?(_), do: false
end

#Implementing the protocol for strings
defimpl Blank, for: BitString do
   def blank?(""), do: true
   def blank?(_), do: false
end

#Implementing the protocol for maps
defimpl Blank, for: Map do
   def blank?(map), do: map_size(map) == 0
end

IO.puts(Blank.blank? [])
IO.puts(Blank.blank? [:true, "Hello"])
IO.puts(Blank.blank? "")
IO.puts(Blank.blank? "Hi")

Вы можете реализовать свой протокол для любого количества или небольшого количества типов, что имеет смысл для использования вашего протокола. Это был довольно простой вариант использования протоколов. Когда вышеуказанная программа запускается, она дает следующий результат -

true
false
true
false

Note - Если вы используете это для любых типов, кроме тех, для которых вы определили протокол, это приведет к ошибке.