Что такое GenServer и какие callbacks он определяет?
GenServer — OTP behaviour для инкапсуляции состояния в процессе. Основные callbacks: init/1 (инициализация), handle_call/3 (синхронный запрос), handle_cast/2 (асинхронный), handle_info/2 (произвольные сообщения), handle_continue/2, terminate/2 и code_change/3.
GenServer: что это и какие callbacks он определяет
GenServer (Generic Server) — это behaviour в OTP, который абстрагирует паттерн клиент-сервер на основе процессов. GenServer инкапсулирует состояние, сериализует доступ к нему и интегрируется с supervision tree. Под капотом это Erlang-процесс с бесконечным циклом receive.
Минимальный пример
defmodule Counter do
use GenServer
# --- Client API ---
def start_link(initial \\ 0) do
GenServer.start_link(__MODULE__, initial, name: __MODULE__)
end
def increment, do: GenServer.cast(__MODULE__, :increment)
def get_value, do: GenServer.call(__MODULE__, :get)
# --- Server Callbacks ---
@impl true
def init(initial) do
{:ok, initial}
end
@impl true
def handle_call(:get, _from, state) do
{:reply, state, state}
end
@impl true
def handle_cast(:increment, state) do
{:noreply, state + 1}
end
@impl true
def handle_info(:reset, _state) do
{:noreply, 0}
end
end
Все callbacks GenServer
init/1
Вызывается при старте процесса. Получает аргумент из start_link/1. Должен вернуть начальное состояние.
@impl true
def init(args) do
# Варианты возврата:
{:ok, initial_state}
{:ok, initial_state, {:continue, :load_data}} # отложенная инициализация
{:ok, initial_state, timeout_ms} # timeout в миллисекундах
:ignore # не запускать процесс
{:stop, reason} # остановить с ошибкой
end
handle_call/3
Синхронный запрос. Клиент блокируется до получения ответа. Три аргумента: запрос, {pid, ref} отправителя, текущий state.
@impl true
def handle_call({:get_user, id}, _from, state) do
user = Map.get(state.users, id)
{:reply, user, state}
# или {:reply, user, new_state, {:continue, :cleanup}}
end
handle_cast/2
Асинхронный запрос. Клиент не ждёт ответа. Используется для операций, результат которых не нужен немедленно.
@impl true
def handle_cast({:add_user, user}, state) do
new_state = Map.put(state, user.id, user)
{:noreply, new_state}
end
handle_info/2
Обрабатывает любые сообщения, отправленные через send/2 или Process.send_after/3, а также системные сообщения (DOWN, EXIT). Это «прочие сообщения».
@impl true
def handle_info(:tick, state) do
# Периодическая задача
Process.send_after(self(), :tick, 1_000)
{:noreply, do_work(state)}
end
@impl true
def handle_info({:DOWN, _ref, :process, pid, reason}, state) do
# Мониторинг упавшего процесса
{:noreply, remove_worker(state, pid)}
end
handle_continue/2
Выполняется сразу после init/1 или другого callback, но до следующего сообщения из queue. Используется для тяжёлой инициализации без блокировки start_link.
@impl true
def init(_args) do
{:ok, %{}, {:continue, :load_from_db}}
end
@impl true
def handle_continue(:load_from_db, state) do
data = Repo.all(MyModel)
{:noreply, %{state | data: data}}
end
terminate/2
Вызывается перед остановкой процесса (если причина не :normal/:shutdown и trap_exit включён). Используется для cleanup.
@impl true
def terminate(reason, state) do
Logger.warning("Counter stopping: #{inspect(reason)}")
:ok
end
code_change/3
Вызывается при горячем обновлении кода через :sys.change_code. Позволяет мигрировать старый state в новую структуру.
@impl true
def code_change(_old_vsn, state, _extra) do
# Мигрировать старый state (map) в новую структуру (struct)
{:ok, struct(NewState, state)}
end
Подводные камни
handle_callблокирует весь GenServer на время выполнения — длинные операции (DB-запросы, HTTP) нужно делегировать в Task, иначе все вызовы встают в очередь.- Не определённый
handle_info/2для всех типов сообщений — неожиданные сообщения накапливаются в inbox и вызывают утечку памяти. - Тяжёлая инициализация в
init/1блокируетstart_linkи, соответственно, весь supervision tree — использоватьhandle_continue. - Использование
GenServer.callиз самого GenServer (рекурсивный вызов) приводит к дедлоку — процесс ждёт ответа от себя. - Забывают передать
name:вstart_link— без имени нельзя обратиться к GenServer по имени, только по PID, который меняется при рестарте. - Дефолтный timeout в
GenServer.call/3— 5 секунд. При медленных операциях нужно явно задать timeout или использовать cast. - Хранение больших данных в state GenServer — они копируются при каждом входящем сообщении, что создаёт нагрузку на GC.
- Игнорирование
@impl true— без него компилятор не предупредит о неверной сигнатуре callback, и функция просто не будет вызвана.
Common mistakes
- Сводить genserver callbacks к названию метода без lifecycle и failure path.
- Игнорировать модель runtime: Elixir компилируется в BEAM bytecode и наследует процессы, message passing, supervision и hot-code friendly модель Erlang VM.
- Не отделять validation, authorization, transaction boundary и business logic.
What the interviewer is testing
- Объясняет genserver callbacks через конкретную точку lifecycle в Elixir.
- Приводит корректный минимальный пример без вымышленных методов или callbacks.
- Называет edge cases: пустые значения, ошибки, транзакции, безопасность или concurrency.