是否可以获取已定义命名空间的列表

2022-08-30 13:40:14

嘿,你好

我想知道在php 5.3 +中是否有一种方法可以获取应用程序中定义的命名空间列表。所以

如果和file 1 has namespace FOOfile 2 has namespace BAR

现在,如果我在文件3 ID中包含文件1和文件2,请通过某种函数调用来了解命名空间FOO和BAR的加载。

我想实现这一点,以确保在检查类是否存在之前加载应用程序中的模块(is_callable)。

如果这是不可能的,我想知道是否有一个函数来检查是否定义了特定的命名空间,比如is_namespace()。

希望你明白这个想法。以及我想要实现的目标


答案 1

首先,要查看类是否存在,请使用class_exists

其次,您可以使用get_declared_classes获取具有命名空间的类列表。

在最简单的情况下,您可以使用它从所有声明的类名中查找匹配的命名空间:

function namespaceExists($namespace) {
    $namespace .= "\\";
    foreach(get_declared_classes() as $name)
        if(strpos($name, $namespace) === 0) return true;
    return false;
}

另一个示例是,以下脚本生成已声明命名空间的分层数组结构:

<?php
namespace FirstNamespace;
class Bar {}

namespace SecondNamespace;
class Bar {}

namespace ThirdNamespace\FirstSubNamespace;
class Bar {}

namespace ThirdNamespace\SecondSubNamespace;
class Bar {}

namespace SecondNamespace\FirstSubNamespace;
class Bar {}

$namespaces=array();
foreach(get_declared_classes() as $name) {
    if(preg_match_all("@[^\\\]+(?=\\\)@iU", $name, $matches)) {
        $matches = $matches[0];
        $parent =&$namespaces;
        while(count($matches)) {
            $match = array_shift($matches);
            if(!isset($parent[$match]) && count($matches))
                $parent[$match] = array();
            $parent =&$parent[$match];

        }
    }
}

print_r($namespaces);

给:

Array
(
    [FirstNamespace] => 
    [SecondNamespace] => Array
        (
            [FirstSubNamespace] => 
        )
    [ThirdNamespace] => Array
        (
            [FirstSubNamespace] => 
            [SecondSubNamespace] => 

        )
)

答案 2

我知道这个问题已经有了答案,但我想为我认为你的问题提供一个更现实的解决方案。如果我昨天有更多的时间发表评论,我会发布这个。很抱歉我没有。

听起来OP有一个模块系统,他需要知道在允许调用特定模块之前是否加载了它。

首先,我想说的是,使用命名空间只是为了声明模块处于活动状态,IMO,滥用它们的用途。如果您遵循名称间距到字母的目的,则您的结构可能看起来更像这样:

整个系统应位于其自己的命名空间中。让我们调用该命名空间 。然后,模块可能位于命名空间下。然后,根据复杂性,每个模块都可能在 下有一个命名空间。以您的示例为例,然后 .SystemSystem\ModuleSystem\ModuleSystem\Module\FOOSystem\Module\BAR

现在,让我们开始制作一个在加载时注册自身的模块系统。

首先,我们需要一个注册的地方。让我们这样称呼它,并且由于可能有很多不同的注册表,因此它将实现 .为简洁起见,我只是发布.很有可能,它还将实现某种全局一致性模型,例如单例,但我也没有展示这一点。在这里:System\Module\RegistrySystem\iRegistrySystem\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 );
        }
    }
); 

另一种可能的方法是为具有特定函数的模块定义一个接口,以便在模块初始化时进行注册。但是,这意味着需要首先加载模块,并可能导致其自身的问题,具体取决于您的需求。

完成此操作后:

  • 你有一个一致的命名空间供模块使用
  • 你有一个一致的接口,允许任何模块知道如何注册自己
  • 将来,您可以轻松扩展模块或注册表接口以获取新内容,同时保持代码干净且易于理解。
  • 而且,最重要的是,您将知道您的模块实际上会声明它们已加载和/或准备就绪,而不是依靠黑魔法为您完成此操作。

推荐