我知道这个问题已经有了答案,但我想为我认为你的问题提供一个更现实的解决方案。如果我昨天有更多的时间发表评论,我会发布这个。很抱歉我没有。
听起来OP有一个模块系统,他需要知道在允许调用特定模块之前是否加载了它。
首先,我想说的是,使用命名空间只是为了声明模块处于活动状态,IMO,滥用它们的用途。如果您遵循名称间距到字母的目的,则您的结构可能看起来更像这样:
整个系统应位于其自己的命名空间中。让我们调用该命名空间 。然后,模块可能位于命名空间下。然后,根据复杂性,每个模块都可能在 下有一个命名空间。以您的示例为例,然后 .System
System\Module
System\Module
System\Module\FOO
System\Module\BAR
现在,让我们开始制作一个在加载时注册自身的模块系统。
首先,我们需要一个注册的地方。让我们这样称呼它,并且由于可能有很多不同的注册表,因此它将实现 .为简洁起见,我只是发布.很有可能,它还将实现某种全局一致性模型,例如单例,但我也没有展示这一点。在这里:System\Module\Registry
System\iRegistry
System\Module\Registry
<?php
namespace System\Module
{
class Registry extends System\Registry
{
protected $registered = array();
public function register( $name=null )
{
$this->registered[] = $name;
}
public function isRegistered( $module )
{
// Code to find module
}
public function getModule( $module )
{
// Code to find module
// OR, if you can't find it...
throw new ModuleNotRegisteredException("Module named \"{$module}\" could not be found in the registry.");
}
}
}
?>
现在,在每个模块中,您需要在加载文件时调用此寄存器函数。有几种方法可以做到这一点。第一种方法是在模块的命名空间中设置一些在负载下运行的代码,类似于典型的程序代码:
namespace System\Module\FOO
{
// Load this module
$system->module->register("FOO");
}
不过,上述情况意味着代码重复。您也可以使用自动加载来代替,这样“注册”代码就全部位于一个位置。这是一个非常基本的概念:
spl_autoload_register(
function ($className)
{
// Code to load files.
// Once loaded register our modules.
if( $namespace = "System\\Module" )
{
$system->module->register( $classname );
}
}
);
另一种可能的方法是为具有特定函数的模块定义一个接口,以便在模块初始化时进行注册。但是,这意味着需要首先加载模块,并可能导致其自身的问题,具体取决于您的需求。
完成此操作后:
- 你有一个一致的命名空间供模块使用
- 你有一个一致的接口,允许任何模块知道如何注册自己
- 将来,您可以轻松扩展模块或注册表接口以获取新内容,同时保持代码干净且易于理解。
- 而且,最重要的是,您将知道您的模块实际上会声明它们已加载和/或准备就绪,而不是依靠黑魔法为您完成此操作。