Анонимные классы

Анонимные классы приносят пользу, когда требуется создать простые одноразовые объекты.

<?php

// Явный класс
class Logger
{
    public function log($msg)
    {
        echo $msg;
    }
}

$util->setLogger(new Logger());

// Анонимный класс
$util->setLogger(new class {
    public function log($msg)
    {
        echo $msg;
    }
});

?>

Анонимные классы по аналогии с обычными классами умеют передавать аргументы в свои конструкторы, расширять другие классы, реализовывать интерфейсы и использовать трейты:

<?php

class SomeClass {}
interface SomeInterface {}
trait SomeTrait {}

var_dump(new class(10) extends SomeClass implements SomeInterface {
    private $num;

    public function __construct($num)
    {
        $this->num = $num;
    }

    use SomeTrait;
});

?>

Результат выполнения приведённого примера:

object(class@anonymous)#1 (1) {
  ["Command line code0x104c5b612":"class@anonymous":private]=>
  int(10)
}

Вложение анонимного класса в другой класс не даёт вложенному классу доступ к закрытым или защищённым методам и свойствам внешнего класса. Анонимному классу потребуется расширить внешний класс, чтобы использовать защищённые свойства и методы внешнего класса. Закрытые свойства внешнего класса потребуется передать в конструктор анонимного класса, чтобы анонимный класс получил к ним доступ:

<?php

class Outer
{
    private $prop = 1;
    protected $prop2 = 2;

    protected function func1()
    {
        return 3;
    }

    public function func2()
    {
        return new class($this->prop) extends Outer {
            private $prop3;

            public function __construct($prop)
            {
                $this->prop3 = $prop;
            }

            public function func3()
            {
                return $this->prop2 + $this->prop3 + $this->func1();
            }
        };
    }
}

echo (new Outer)->func2()->func3();

Результат выполнения приведённого примера:

6

Объекты, которые создаёт одно и то же объявление анонимного класса, относятся к экземплярам одного и того же класса.

<?php

function anonymous_class()
{
    return new class {};
}

if (get_class(anonymous_class()) === get_class(anonymous_class())) {
    echo 'Тот же класс';
} else {
    echo 'Другой класс';
}

?>

Результат выполнения приведённого примера:

Тот же класс

Замечание:

Обратите внимание, что движок присваивает анонимным классам названия, как показывает следующий пример. Это название рекомендуют рассматривать только как деталь реализации, и лучше на него не полагаться.

<?php

echo get_class(new class {});

?>

Вывод приведённого примера будет похож на:

class@anonymous/in/oNi1A0x7f8636ad2021

Анонимные классы только для чтения

Начиная с PHP 8.3.0 к анонимным классам разрешили применять модификатор readonly.

Пример #1 Пример определения анонимного класса только для чтения

<?php

// Определяем анонимный класс доступным только для чтения и передаём как аргумент
var_dump(new readonly class('[DEBUG]') {
    public function __construct(private string $prefix) {}

    public function log($msg)
    {
        echo $this->prefix . ' ' . $msg;
    }
});

?>