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

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

<?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;
}
});

?>