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.

Sources

Related topics