У меня к вам абстрактный вопрос.
Вопрос: Как может подкласс, расширяющий абстрактный класс, зарегистрировать себя до абстрактного класса или другого класса?
Проблема:
Думаю, у нас есть мастер модуля с именем ModuleMaster, и, возможно, кто-то другой напишет другие модули для решения конкретной проблемы, не изменяя мастер-класс, и назвал его ModuleA. По этой причине мы хотим реализовать динамическую загрузку решений проблем.
Моя идея:
Файл: Extensions.php:
namespace Project\Extensions;
class Extensions
{
public function getLoadedModules()
{
var_dump(ModuleMaster::LOADED_MODULES);
}
}
Файл: Modules \ ModuleMaster.php:
namespace Project\Extensions\Modules;
abtract class ModuleMaster
{
public const LOADED_MODULES = array();
}
Файл: Modules \ ModuleA.php:
namespace Project\Extensions\Modules;
class ModuleA extends ModuleMaster
{
}
Надеюсь, вы понимаете, что я имею в виду, и можете помочь с этой абстрактной проблемой.






Это очень странно использовать. Вероятно, лучше иметь внешний реестр для вашего модуля. Но я думаю, вы просите об этом:
Файл: Extensions.php:
namespace Project\Extensions;
use \Project\Extensions\Modules\ModuleMaster;
class Extensions
{
public function getLoadedModules()
{
var_dump(ModuleMaster::getLoadedModules());
}
}
Файл: Modules \ ModuleMaster.php:
namespace Project\Extensions\Modules;
abstract class ModuleMaster
{
public static function getLoadedModules() {
$parent = self::class;
return array_values(array_filter(\get_declared_classes(), function ($class) use ($parent) {
return in_array($parent, class_parents($class));
}));
}
}
Файл: Modules \ ModuleA.php:
namespace Project\Extensions\Modules;
use \Project\Extensions\Modules\ModuleMaster;
class ModuleA extends ModuleMaster
{
}
Пример использования:
$e = new \Project\Extensions\Extensions;
$e->getLoadedModules();
Пример результата:
array(1) {
[0]=>
string(34) "Project\Extensions\Modules\ModuleA"
}
Обратите внимание, что код работает только в том случае, если все ваши файлы классов включены в контекст перед запуском getLoadedModules(). PHP не будет знать, что ваш класс существует, если он еще не загружен в контекст.
Кажется, вы пытаетесь создать возможность в родительском классе, которая не требуется или не подходит для дочернего класса. Это противоположность наследования и, следовательно, антипаттерн. Кроме того, даже если это можно рассматривать как расширение отражения, вы пытаетесь поместить данные времени выполнения в класс - это не то, для чего предназначены классы.
Вы также никоим образом не объяснили, что я могу понять Почему, что вы хотите это сделать.
Я подозреваю, что вы действительно хотите реализовать фабрику, стратегию или объект реестра.