Автозагрузка PHP в пространствах имен - PullRequest
11 голосов
/ 04 сентября 2010

У меня была небольшая проблема с автозагрузкой в ​​моем пространстве имен. Как показано в руководстве по PHP здесь: http://us.php.net/manual/en/language.namespaces.rules.php, вы должны иметь возможность автоматически загружать функции пространства имен с полным именем, например. \ Клей \ Common \ 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 ]

14 голосов
/ 04 сентября 2010

Вот единственный правильный ответ.

Каждое пространство имен нуждается в собственной функции spl_autoload_register ().

также, синтаксис spl_autoload_register () изменено в 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);

Вот некоторый живой код, который просто работает!

in ../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.

Так что я знаю, что это работает.

2 голосов
/ 26 декабря 2017

Используйте Composer для автозагрузки ваших классов PHP.

Узнайте, как это сделать, в моем недавнем сообщении в блоге: https://enchanterio.github.io/enterprise-level-php/2017/12/25/the-magic-behind-autoloading-php-files-using-composer.html

1 голос
/ 11 мая 2017

Неправильное представление в вопросе OP, вероятно, заключается в том, что функции / методы будут подвергаться автозагрузке, а это не так.Автозагрузка вызывается только ссылками на классы.

При этом все еще остается вопрос об автозагрузке классов в пространствах имен :

По состоянию на 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 .

Пример кода выше (и еще один для автозагрузки из нескольких пространств имен) можно найти по адресу Пример реализации PSR-4 (или GitHub: fig-standards / принято / PSR-4-autoloader-examples.md * * тысячу двадцать-один). * +1022 *

...