PHPMiddleCoding
Что такое nullsafe operator (?->) в PHP 8 и когда его использовать?
Оператор ?-> (PHP 8.0) прерывает вычисление цепочки вызовов при null и возвращает null вместо фатальной ошибки. Применяйте для опциональных объектов в цепочках; не используйте там, где null — признак программной ошибки.
Nullsafe operator (?->) в PHP 8
Оператор ?-> введён в PHP 8.0. Он вычисляет левую часть и, если она равна null, прекращает вычисление всей цепочки и возвращает null вместо выброса Error: Call to a member function on null.
Проблема до PHP 8
<?php
// Получить город пользователя из заказа
// Каждый объект в цепочке может быть null
$city = null;
if ($order !== null) {
$user = $order->getUser();
if ($user !== null) {
$address = $user->getAddress();
if ($address !== null) {
$city = $address->getCity();
}
}
}
// Или через тернарный — не лучше
$city = $order
? ($order->getUser()
? ($order->getUser()->getAddress()
? $order->getUser()->getAddress()->getCity()
: null)
: null)
: null;
Решение с ?->
<?php
// Та же логика — одна строка
$city = $order?->getUser()?->getAddress()?->getCity();
// Если $order == null → $city = null (без ошибок)
// Если getUser() вернул null → $city = null
// Если getAddress() вернул null → $city = null
// Иначе → $city = строка из getCity()
Расширенные примеры использования
<?php
// Доступ к свойству
$name = $user?->profile?->name;
// Вызов метода с аргументами
$formatted = $user?->getAddress()?->format(short: true);
// Комбинирование с ?? (null-коалесцент)
$city = $order?->getUser()?->getAddress()?->getCity() ?? 'Unknown';
// Доступ к массиву через ?-> не работает:
// $val = $obj?->items['key']; // не защищает items от null
// Правильно:
$val = $obj?->getItems()['key'] ?? null;
// Со статическими методами — НЕ работает
// $result = $obj?->MyClass::staticMethod(); // синтаксическая ошибка
Nullsafe с new и first-class callables
<?php
// ?-> работает на результате new
function getRepository(): ?UserRepository
{
return null; // или объект
}
$user = getRepository()?->find(1);
// В PHP 8.1+ с First-class callables:
$finder = getRepository()?->find(...);
// $finder будет Closure или null
Когда НЕ использовать ?->
<?php
// 1. Когда null — ошибка программирования, а не ожидаемое состояние
// Если $service не может быть null — не скрывайте проблему
class OrderProcessor
{
public function __construct(
private readonly PaymentService $paymentService // Инъекция обязательна
) {}
public function process(Order $order): void
{
// НЕ пишите $this->paymentService?->charge()
// Если paymentService null — это баг DI-контейнера
$this->paymentService->charge($order->getTotal());
}
}
// 2. Когда нужно различать «null пришёл» vs «метод вернул null»
$result = $api?->fetchData();
if ($result === null) {
// Нельзя понять: $api был null или fetchData() вернул null?
}
Подводные камни
- Короткое замыкание всей цепочки — если в середине цепочки встречается null, все последующие вызовы пропускаются. Побочные эффекты после null не выполнятся:
$obj?->save()?->notify()— notify не вызовется, если save вернул null. - Не защищает доступ к массивам —
$obj?->data['key']защищает только разыменование$obj, но еслиdata— null, будет ошибка при доступе по ключу. - Не работает для статических вызовов —
$obj?->ClassName::method()вызовет синтаксическую ошибку; nullsafe применим только к методам экземпляра. - Маскировка архитектурных проблем — избыточное использование ?-> часто признак плохого дизайна. Null Object Pattern или строгие типы могут быть лучшим решением.
- Несовместимость с PHP 7 — если проект поддерживает PHP 7.x, оператор нельзя использовать; нужна явная проверка или пакет-полифилл невозможен (синтаксическая конструкция).
- Запись через ?-> не работает —
$obj?->property = 'value'вызовет fatal error в PHP 8.0; присваивание через nullsafe не поддерживается. - Отладка null-цепочек — при длинной цепочке трудно определить, на каком шаге вернулся null; разбейте на переменные при отладке.
Common mistakes
- Сводить nullsafe operator к названию метода без lifecycle и failure path.
- Игнорировать модель runtime: интерпретируемый runtime PHP 8.x, обычно запускаемый как отдельный запрос в FPM, CLI или worker-процессе.
- Не отделять validation, authorization, transaction boundary и business logic.
What the interviewer is testing
- Объясняет nullsafe operator через конкретную точку lifecycle в PHP.
- Приводит корректный минимальный пример без вымышленных методов или callbacks.
- Называет edge cases: пустые значения, ошибки, транзакции, безопасность или concurrency.