确定调用函数的命名空间

vol*_*zta 7 php oop namespaces

我想知道在调用函数时是否可以确定当前命名空间是什么.我有这个函数声明:

<?php
namespace Site\Action;
function add ($hook, $function) {
    /**
     * determine the namespace this was called from because
     * __NAMESPACE__ is "site\action" :(
     */
     if (is_callable($function))
         call_user_func($function);
}
?>
Run Code Online (Sandbox Code Playgroud)

在另一个文件上:

<?php
namespace Foo;
function bar () {
}
?>
Run Code Online (Sandbox Code Playgroud)

让我们说我把它作为我的程序代码:

<?php
namespace Foo;
Site\Action\add('hookname', 'bar');
?>
Run Code Online (Sandbox Code Playgroud)

假设Bar在这种情况下打算解决,Foo\bar因为那是从它调用的命名空间.

这是一个很长的解释,是否有可能确定Site\Action\add()调用的活动命名空间?

提前致谢.

nic*_*kl- 4

您正在寻找的是: ReflectionFunctionAbstract::getNamespaceName

如果你想知道你来自哪里,debug_backtrace()就是你的朋友。

以下应该可以解决您的困惑:

function backtrace_namespace() 
{
    $trace = array();
    $functions = array_map(
        function ($v) {
            return $v['function'];
        },
        debug_backtrace()
    );
    foreach ($functions as $func) {
        $f = new ReflectionFunction($func);
        $trace[] = array(
            'function' => $func, 
            'namespace' =>  $f->getNamespaceName()
        );
    }
    return $trace;
}
Run Code Online (Sandbox Code Playgroud)

只需从任何地方调用它即可查看回溯。我修改了您的“程序”代码文件如下:

namespace Foo;

function bar () 
{
    var_export(backtrace_namespace());
}

/** The unasked question: We need to use the fully qualified name currently. */
function go() 
{
    \Site\Action\add('hookname', 'Foo\\bar');
}

go();
Run Code Online (Sandbox Code Playgroud)

包含此文件的结果将在标准输出上如下:

array (
    0 =>
    array (
        'function' => 'backtrace_namespace',
        'namespace' => '',
    ),
    1 =>
    array (
        'function' => 'Foo\\bar',
        'namespace' => 'Foo',
    ),
    2 =>
    array (
        'function' => 'call_user_func',
        'namespace' => '',
    ),
    3 =>
    array (
        'function' => 'Site\\Action\\add',
        'namespace' => 'Site\\Action',
    ),
    4 =>
    array (
        'function' => 'Foo\\go',
        'namespace' => 'Foo',
    ),
)
Run Code Online (Sandbox Code Playgroud)

现在为了加分,隐藏问题的答案:

如何解析调用命名空间以避免使用完全限定的函数名称作为参数?

以下内容将允许您按预期调用该函数:

 Site\Action\add('hookname', 'bar');
Run Code Online (Sandbox Code Playgroud)

没有得到可怕的:

警告:call_user_func() 期望参数 1 是有效的回调,未找到函数“bar”或函数名称无效

因此,在重新设计之前,请尝试一下尺寸:

namespace Site\Action;

function add($hook, $function) 
{
    $trace = backtrace_namespace();
    $prev = (object) end($trace);

    $function = "$prev->namespace\\$function";

    if (is_callable($function))
        call_user_func($function);
}
Run Code Online (Sandbox Code Playgroud)

我认为没有理由debug_backtrace不应该使用,这就是它的用途。

欢乐!