Эликсир - Процессы

В Elixir весь код выполняется внутри процессов. Процессы изолированы друг от друга, выполняются параллельно друг другу и обмениваются сообщениями. Не следует путать процессы Elixir с процессами операционной системы. Процессы в Elixir чрезвычайно легкие с точки зрения памяти и ЦП (в отличие от потоков во многих других языках программирования). Из-за этого нередко одновременно работают десятки или даже сотни тысяч процессов.

В этой главе мы узнаем об основных конструкциях для порождения новых процессов, а также об отправке и получении сообщений между различными процессами.

Функция появления

Самый простой способ создать новый процесс - использовать spawnфункция. Вspawnпринимает функцию, которая будет запущена в новом процессе. Например -

pid = spawn(fn -> 2 * 2 end)
Process.alive?(pid)

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

false

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

Как уже упоминалось, все коды Elixir выполняются внутри процессов. Если вы запустите функцию self, вы увидите PID для вашего текущего сеанса -

pid = self
 
Process.alive?(pid)

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

true

Сообщение передается

Мы можем отправлять сообщения процессу с send и получить их с receive. Давайте передадим сообщение текущему процессу и получим его тем же.

send(self(), {:hello, "Hi people"})

receive do
   {:hello, msg} -> IO.puts(msg)
   {:another_case, msg} -> IO.puts("This one won't match!")
end

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

Hi people

Мы отправили сообщение текущему процессу с помощью функции отправки и передали его PID self. Затем мы обработали входящее сообщение, используяreceive функция.

Когда сообщение отправляется процессу, оно сохраняется в process mailbox. Блок приема проходит через почтовый ящик текущего процесса в поисках сообщения, которое соответствует любому из заданных шаблонов. Блок приема поддерживает охранники и многие пункты, такие как case.

Если в почтовом ящике нет сообщений, соответствующих какому-либо шаблону, текущий процесс будет ждать, пока не прибудет соответствующее сообщение. Также можно указать тайм-аут. Например,

receive do
   {:hello, msg}  -> msg
after
   1_000 -> "nothing after 1s"
end

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

nothing after 1s

NOTE - Таймаут 0 может быть задан, если вы уже ожидаете, что сообщение будет в почтовом ящике.

Ссылки

Самая распространенная форма появления в Эликсире - через spawn_linkфункция. Прежде чем взглянуть на пример с spawn_link, давайте разберемся, что происходит, когда процесс выходит из строя.

spawn fn -> raise "oops" end

Когда вышеуказанная программа запускается, она выдает следующую ошибку -

[error] Process #PID<0.58.00> raised an exception
** (RuntimeError) oops
   :erlang.apply/2

Произошла ошибка, но процесс создания все еще продолжается. Это потому, что процессы изолированы. Если мы хотим, чтобы сбой в одном процессе распространялся на другой, нам нужно связать их. Это можно сделать с помощьюspawn_linkфункция. Давайте рассмотрим пример, чтобы понять то же самое -

spawn_link fn -> raise "oops" end

Когда вышеуказанная программа запускается, она выдает следующую ошибку -

** (EXIT from #PID<0.41.0>) an exception was raised:
   ** (RuntimeError) oops
      :erlang.apply/2

Если вы запускаете это в iexshell, то оболочка обрабатывает эту ошибку и не завершает работу. Но если вы запустите, сначала создав файл сценария, а затем используяelixir <file-name>.exs, родительский процесс также будет остановлен из-за этого сбоя.

Процессы и ссылки играют важную роль при построении отказоустойчивых систем. В приложениях Elixir мы часто связываем наши процессы с супервизорами, которые обнаруживают, когда процесс умирает, и запускают новый процесс вместо него. Это возможно только потому, что процессы изолированы и по умолчанию ничего не разделяют. А поскольку процессы изолированы, сбой в процессе не может привести к сбою или повреждению состояния другого. В то время как другие языки потребуют от нас перехвата / обработки исключений; в Elixir мы вполне можем позволить процессам выйти из строя, потому что мы ожидаем, что супервизоры правильно перезапустят наши системы.

состояние

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

Мы можем писать процессы, которые зацикливаются бесконечно, поддерживают состояние и отправляют и получают сообщения. В качестве примера давайте напишем модуль, который запускает новые процессы, которые работают как хранилище ключей и значений в файле с именемkv.exs.

defmodule KV do
   def start_link do
      Task.start_link(fn -> loop(%{}) end)
   end

   defp loop(map) do
      receive do
         {:get, key, caller} ->
         send caller, Map.get(map, key)
         loop(map)
         {:put, key, value} ->
         loop(Map.put(map, key, value))
      end
   end
end

Обратите внимание, что start_link функция запускает новый процесс, который запускает loopфункция, начиная с пустой карты. ВloopЗатем функция ожидает сообщений и выполняет соответствующее действие для каждого сообщения. В случае:getmessage, он отправляет сообщение обратно вызывающему абоненту и снова вызывает цикл, чтобы дождаться нового сообщения. В то время как:put сообщение действительно вызывает loop с новой версией карты с сохраненным заданным ключом и значением.

Давайте теперь запустим следующее -

iex kv.exs

Теперь ты должен быть в своем iexоболочка. Чтобы протестировать наш модуль, попробуйте следующее -

{:ok, pid} = KV.start_link

# pid now has the pid of our new process that is being 
# used to get and store key value pairs 

# Send a KV pair :hello, "Hello" to the process
send pid, {:put, :hello, "Hello"}

# Ask for the key :hello
send pid, {:get, :hello, self()}

# Print all the received messages on the current process.
flush()

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

"Hello"