Синтаксис callable первого класса представлен в PHP 8.1.0 как способ создания анонимных функций из callable . Он заменяет существующий вызываемый синтаксис с использованием строк и массивов. Преимущество этого синтаксиса в том, что он доступен для статического анализа и использует область действия в точке, где получен вызываемый объект.

CallableExpr(...)синтаксис используется для создания объекта Closure из callable. CallableExprпринимает любое выражение, которое может быть напрямую вызвано в грамматике PHP:

Пример #1 Простой вызываемый синтаксис первого класса

<?php

class Foo {
   public function method() {}
   public static function staticmethod() {}
   public function __invoke() {}
}

$obj = new Foo();
$classStr = 'Foo';
$methodStr = 'method';
$staticmethodStr = 'staticmethod';


$f1 = strlen(...);
$f2 = $obj(...);  // invokable object
$f3 = $obj->method(...);
$f4 = $obj->$methodStr(...);
$f5 = Foo::staticmethod(...);
$f6 = $classStr::$staticmethodStr(...);

// traditional callable using string, array
$f7 = 'strlen'(...);
$f8 = [$obj, 'method'](...);
$f9 = [Foo::class, 'staticmethod'](...);
?>

Примечание :

...Является частью синтаксиса, а не упущением .

CallableExpr(...)имеет ту же семантику, что и Closure::fromCallable() . То есть, в отличие от callable, использующего строки и массивы, CallableExpr(...)учитывает область действия в точке, где она создается:

Пример #2 Сравнение области действия CallableExpr(...)и традиционного callable

<?php

class Foo {
    public function getPrivateMethod() {
        return [$this, 'privateMethod'];
    }

    private function privateMethod() {
        echo __METHOD__, "\n";
    }
}

$foo = new Foo;
$privateMethod = $foo->getPrivateMethod();
$privateMethod();
// Fatal error: Call to private method Foo::privateMethod() from global scope
// This is because call is performed outside from Foo and visibility will be checked from this point.

class Foo1 {
    public function getPrivateMethod() {
        // Uses the scope where the callable is acquired.
        return $this->privateMethod(...); // identical to Closure::fromCallable([$this, 'privateMethod']);
    }

    private function privateMethod() {
        echo __METHOD__, "\n";
    }
}

$foo1 = new Foo1;
$privateMethod = $foo1->getPrivateMethod();
$privateMethod();  // Foo1::privateMethod
?>

Примечание :

Создание объекта с помощью этого синтаксиса (например new Foo(...), ) не поддерживается, поскольку new Foo()синтаксис не считается вызовом.

Примечание :

Первоклассный синтаксис callable нельзя комбинировать с оператором nullsafe . Оба следующих результата приводят к ошибке времени компиляции:

<?php
$obj?->method(...);
$obj?->prop->method(...);
?>