首页 文章

PHP在命名空间中自动加载

提问于
浏览
11

我的命名空间中的自动加载有点问题 . 如PHP手册所示:http://us.php.net/manual/en/language.namespaces.rules.php您应该能够使用完全限定名称自动加载命名空间函数,例如\胶\共同\ is_email() .

事情是我有一个函数spl_autoload_register(array($ import,“load”));在初始命名空间内,但每当我尝试从初始命名空间调用\ glue \ common \ is_email()时,它都不会传递自动加载功能,但是当使用新的is_email()时(在类的上下文中)它将会 . 我不明白手册说我可以从完全合格的名字自动加载,但我不能: .

这是我的代码:

namespace glue;

require_once 'import.php';

use glue\import as import;
use glue\core\router as router;

$import = new import();

spl_autoload_register(array($import, "load"));

/** Works and echos glue\router **/
$router = new router();

/** Don't do nothing **/
$cheese = \glue\common\is_email($email);

我也试过这段代码:

namespace glue;

require_once 'import.php';

use glue\import as import;
use glue\core\router as router;
use glue\common;

$import = new import();

spl_autoload_register(array($import, "load"));

/** Works and echos glue\router **/
$router = new router();

/** Don't do nothing **/
$cheese = common\is_email($email);

最后这段代码:

namespace glue;

require_once 'import.php';

use glue\import as import;
use glue\core\router as router;
use glue\common\is_email as F;

$import = new import();

spl_autoload_register(array($import, "load"));

/** Works and echos glue\router **/
$router = new router();

/** Don't do nothing **/
$cheese = F($email);

3 回答

  • 2

    这是唯一正确的答案 .

    Every namespace needs its own spl_autoload_register() function.

    还, spl_autoload_register() syntax changed in 5.3:

    spl_autoload_register(__NAMESPACE__ . "\\className::functionName"));
    

    以下应该有效:

    namespace glue;
    
    require_once 'import.php';
    
    use glue\import as import;
    use glue\core\router as router;
    
    $import = new import();
    
    spl_autoload_register(__NAMESPACE__ . "\\$import::load"));
    
    /** Works and echos glue\router **/
    $router = new router();
    
    /** Don't do nothing **/
    $cheese = \glue\common\is_email($email);
    

    这是一些正常工作的 live 代码!

    在../WebPageConsolidator.inc.php中:

    class WebPageConsolidator
    {
        public function __construct() { echo "PHP 5.2 constructor.\n"; }
    }
    

    在test.php中:

    <?php
    
    namespace WebPage;
    
    class MyAutoloader
    {
        public static function load($className)
        {
            require '../' . __NAMESPACE__ . $className . '.inc.php';
        }
    }
    
    spl_autoload_register(__NAMESPACE__ . "\\MyAutoloader::load");
    
    class Consolidator extends \WebpageConsolidator
    {
        public function __construct()
        {
            echo "PHP 5.3 constructor.\n";
    
            parent::__construct();
        }
    }
    
    // Output: 
    // PHP 5.3 constructor.
    // PHP 5.2 constructor.
    

    所以我知道它有效 .

  • 13

    使用Composer自动加载您的PHP类 .

    在我最近的博客文章中查看如何操作:https://enchanterio.github.io/enterprise-level-php/2017/12/25/the-magic-behind-autoloading-php-files-using-composer.html

  • 0

    OP问题中的误解可能是函数/方法会受到自动加载 - 它们不是 . 自动加载仅由引用类触发 .

    据说这仍然是 question about autoloading classes in namespaces

    截至2017年,自动加载的当前PHP-FIG标准是PSR-4,它为命名空间类提供以下自动加载器代码:

    <?php
    /**
     * An example of a project-specific implementation.
     *
     * After registering this autoload function with SPL, the following line
     * would cause the function to attempt to load the \Foo\Bar\Baz\Qux class
     * from /path/to/project/src/Baz/Qux.php:
     *
     *      new \Foo\Bar\Baz\Qux;
     *
     * @param string $class The fully-qualified class name.
     * @return void
     */
    spl_autoload_register(function ($class) {
    
        // project-specific namespace prefix
        $prefix = 'Foo\\Bar\\';
    
        // base directory for the namespace prefix
        $base_dir = __DIR__ . '/src/';
    
        // does the class use the namespace prefix?
        $len = strlen($prefix);
        if (strncmp($prefix, $class, $len) !== 0) {
            // no, move to the next registered autoloader
            return;
        }
    
        // get the relative class name
        $relative_class = substr($class, $len);
    
        // replace the namespace prefix with the base directory, replace namespace
        // separators with directory separators in the relative class name, append
        // with .php
        $file = $base_dir . str_replace('\\', '/', $relative_class) . '.php';
    
        // if the file exists, require it
        if (file_exists($file)) {
            require $file;
        }
    });
    

    完整的规范文本可以在PSR-4: Autoloader找到 .

    上面的代码示例(以及另一个来自多个名称空间的自动加载)可以在Example Implementations of PSR-4(或GitHub:fig-standards/accepted/PSR-4-autoloader-examples.md)找到 .

相关问题