php autoloader:为什么这个工作?

Eri*_*ler 5 php spl-autoloader

我一直在尝试自动加载器类目录映射技术,这有点困难.我设法提出了一个相当直接的解决方案(表面上看),但我完全神秘的是它完全有效,而其他"更明显"的解决方案失败了.下面是一些代码片段,说明了我的困惑.

这是工作代码:

<?php
    spl_autoload_register('my_autoloader');

    function my_autoloader($class) {
        $classMap = array(
            'classes/',
            'classes/sites/',
            'classes/data/'
        );
        foreach ($classMap as $location) {
            if (!@include_once($location . $class . '.php')) { // @ SUPPRESSES include_once WARNINGS
                // CLASS DOESN'T EXIST IN THIS DIRECTORY
                continue;
            } else {
                // CLASS IS AUTO-LOADED
                break;
            }
        }
    }
?>
Run Code Online (Sandbox Code Playgroud)

这是我觉得应该工作的片段,但不是:

<?php
    spl_autoload_register('my_autoloader');

    function my_autoloader($class) {
        $classMap = array(
            'classes/',
            'classes/sites/',
            'classes/data/'
        );
        foreach ($classMap as $location) {
            if (file_exists($location . $class . '.php')) {
                require_once ($location . $class . '.php');
            }
        }
    }
?>
Run Code Online (Sandbox Code Playgroud)

后者对我更有意义,因为虽然这两个版本有效:

<?php
    spl_autoload_register('my_autoloader');

    function my_autoloader($class) {
        require_once ('classes/sites/' . $class . '.php');
    }
?>

<?php
    spl_autoload_register('my_autoloader');

    function my_autoloader($class) {                
        $location = 'classes/sites/';                
        require_once ($location . $class . '.php');
    }
?>
Run Code Online (Sandbox Code Playgroud)

这个抛出"没有这样的文件或目录......"(注意路径中缺少"sites /").

<?php
    spl_autoload_register('my_autoloader');

    function my_autoloader($class) {
        require_once ('classes/' . $class . '.php');
    }
?>
Run Code Online (Sandbox Code Playgroud)

"没有这样的文件或目录..."错误让我觉得我可以简单地检查一个类的支持文件,if(file_exists()){require_once(); break;} else {continue;}

为什么不起作用?而且,为什么第一个代码段工作?永远不会明确包含或要求支持路径/文件.

Eri*_*ler 1

好吧,我明白了。我的问题确实是没有正确设置路径;使用__DIR__常量是::ahem:: 成功之路。这是我现在使用的工作代码:

<?php
    spl_autoload_register('my_autoloader');

    function my_autoloader($class) {
        $classMap = array(
            'classes/',
            'classes/sites/',
            'classes/data/'
        );
        foreach ($classMap as $location) {
            if (file_exists(__DIR__ . '/' . $location . $class . '.php')) {
                require_once(__DIR__ . '/' . $location . $class . '.php');
                break;
            }
        }
    }
?>
Run Code Online (Sandbox Code Playgroud)

事实证明,它__DIR__返回调用它的实际文件的目录位置(而不是包含它的文件),这意味着只要该文件保留在带有 my 的目录的根目录中,它就可以工作。 /classes 目录。而且,它的存在专门用于定义这些类型的设置......

希望这可以帮助其他人!当然,如果有人能解释为什么这个解决方案不是最优的,我会全神贯注......

编辑:我将执行的一项优化是将 $classMap array() 转换为关联数组(例如,。$classMap = array('root' => 'classes/', 'sites' => 'classes/sites/');这样我可以进行查找,而不是循环遍历我随时间创建的所有目录。但是,我想这是另一个线。

EDIT2:如果有人感兴趣,这是我想出的将其作为关联数组执行的解决方案。基本上,我在 $GLOBALS 中设置了一个关联数组,并使用它来存储类 -> 包的映射。然后,在自动加载器中,我映射 Packages -> Locations,然后调用实例化类所需的文件。

这是全局配置文件:

<?php
    // INITIALIZE GLOBAL Class -> Package map
    if (!array_key_exists('packageMap',$GLOBALS)) {
        $GLOBALS['packageMap'] = array();
    }

    spl_autoload_register('my_autoloader');

    function my_autoloader($class) {
        $classMap = array(
            'root'=>'/classes/',
            'sites'=>'/classes/sites/',
            'data'=>'/classes/data/'
        );

        // RESOLVE MAPPINGS TO AN ACTUAL LOCATION
        $classPackage = $GLOBALS['packageMap'][$class];
        $location = $classMap[$classPackage];
        if (file_exists(__DIR__ . $location . $class . '.php')) {
            require_once(__DIR__ . $location . $class . '.php');
        }
    }
?>
Run Code Online (Sandbox Code Playgroud)

这是使用自动加载器的特定站点配置文件:

<?php   
    // LOAD GLOBAL CONFIG FILE      
    require_once('../config/init.php');

    // CREATE LOCAL REFERENCE TO GLOBAL PACKAGE MAP (MOSTLY FOR READABILITY)
    $packageMap = $GLOBALS['packageMap'];

    // ADD Class -> Package MAPPINGS
    $packageMap['DAO'] = 'data';
    $packageMap['SomeSite'] = 'sites';
    $packageMap['PDOQuery'] = 'data';

    // INSTANTIATE SOMETHING IN ONE OF THE PACKAGES
    $someSite = new SomeSite();
?>
Run Code Online (Sandbox Code Playgroud)

希望这对其他人有用...如果它对任何人提出任何危险信号,请插话。最终,我想用 apc 替换 $GLOBALS 的使用,但它没有安装在我的托管服务器上:/。我可能会考虑 memcached,但我听到的评论褒贬不一......