Анонимные классы приносят пользу, когда требуется создать простые одноразовые объекты.
<?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;
}
});
?>