如何使用 __autoload 从多个目录加载类?

Posted

技术标签:

【中文标题】如何使用 __autoload 从多个目录加载类?【英文标题】:How can I load classes from multiple directories with __autoload? 【发布时间】:2011-12-04 12:17:55 【问题描述】:

跟进此question,似乎只需使用下面的__autoload 代码即可解决重复问题,

function __autoload($class_name) 

    include AP_SITE."classes_1/class_".$class_name.".php";


$connection = new database_pdo(DSN,DB_USER,DB_PASS);
var_dump($connection);

结果,

object(database_pdo)[1]
  protected 'connection' => 
    object(PDO)[2]

但是这只从一个目录加载类,其他目录呢?因为我将类分组在不同的目录中。所以如果我想从其他目录加载类,我会得到错误,

function __autoload($class_name) 

    include AP_SITE."classes_1/class_".$class_name.".php";
    include AP_SITE."classes_2/class_".$class_name.".php";

消息,

警告: 包括(C:/wamp/www/art_on_your_doorstep_2011_MVC/global/applications/CART/classes_2/class_database_pdo.php) [function.include]:无法打开流:没有这样的文件或目录 在...

指的是这一行 - include AP_SITE."classes_2/class_".$class_name.".php";

所以,我的问题是 - 如何使用 __autoload 从多个目录加载类?

一种可能的解决方案:

function autoload_class_multiple_directory($class_name) 


    # List all the class directories in the array.
    $array_paths = array(
        'classes_1/', 
        'classes_2/'
    );

    # Count the total item in the array.
    $total_paths = count($array_paths);

    # Set the class file name.
    $file_name = 'class_'.strtolower($class_name).'.php';

    # Loop the array.
    for ($i = 0; $i < $total_paths; $i++) 
    
        if(file_exists(AP_SITE.$array_paths[$i].$file_name)) 
        
            include_once AP_SITE.$array_paths[$i].$file_name;
         
    


spl_autoload_register('autoload_class_multiple_directory');

【问题讨论】:

【参考方案1】:

您可以使用spl_autoload_register 而不是单个__autoload 函数来注册多个自动加载函数。这是推荐的方式。

如果一个自动加载器能够加载文件,堆栈中的下一个将不会被调用。

然而,每个自动加载器都应该只加载它所针对的类,因此您需要通过类名和/或is_file 进行检查。使用类名通常会更好,因为如果您的应用程序增长,在文件系统上疯狂尝试可能会给系统带来压力。

为了不重新发明***,您甚至可以使用已经存在的自动加载器,它能够处理文件名调用的 PSR-0 标准。这些通常允许在基本目录上注册特定的命名空间。在您的情况下,这意味着您必须根据PSR-0 convention 重命名和组织您的文件。


快速解决方案(与您的问题有关):

function __autoload($class_name) 

    $file = sprintf('%sclasses_1/class_%s.php', AP_SITE, $class_name);
    if (is_file($file))
    
        include $file;
        return;
    
    $file = sprintf('%sclasses_2/class_%s.php', AP_SITE, $class_name);
    if (is_file($file))
    
        include $file;
        return;
    

如您所见,已经有重复的代码(和您的一样)。因此,这应该只是一个临时解决方案,因为您最终会为要测试的每个目录获得越来越多的重复行。如果您考虑更改设计,请考虑 PSR-0 shema,它有助于简化代码库并便于重用 PHP 世界中的其他现有组件。


function autoload_class_multiple_directory($class_name) 


    # List all the class directories in the array.
    $array_paths = array(
        'classes_1/', 
        'classes_2/'
    );

    foreach($array_paths as $path)
    
        $file = sprintf('%s%s/class_%s.php', AP_SITE, $path, $class_name);
        if(is_file($file)) 
        
            include_once $file;
         

    


spl_autoload_register('autoload_class_multiple_directory');

【讨论】:

感谢 hakre,是的,当我有很多目录时,我可以长期看到这个解决方案的问题。 我想出了另一个想法 - 请参阅上面的编辑/可能的解决方案 :) @lauthiamkok:是的,我之前也想过这个问题,但想留点空间。查看我之前想到的解决方案,我现在添加了它。请参阅foreach,如果您将文件名分配给变量 ($file),您可以更好地重复使用它。 请问-%s%s%s 是什么意思? 是 sprintf 中字符串值的占位符。还有更多,您可以在php.net/manual/en/function.sprintf.php 中阅读更多相关信息。我通常使用 sprintf 而不是在字符串中将变量合并在一起,因为它可以进行更细粒度的控制,并且您可以在格式化字符串之后看到所有变量作为它自己的参数。【参考方案2】:

关注 PSR-0

正确的做法是采用PSR-0 naming convention,然后使用兼容PSR-0的自动加载器,例如Symfony2的ClassLoader组件中的UniversalClassLoader

例如:

a/src/ProjectA/Database/Pdo.php

<?php

namespace ProjectA\Database;

class Pdo

    // your code

b/src/ProjectB/Mail/Smtp.php

<?php

namespace ProjectB\Mail;

class Smtp

    // your code

symfony ClassLoader 在vendor/symfony/src/Symfony/Component/ClassLoader

autoload.php

<?php

require __DIR__.'/vendor/symfony/src/Symfony/Component/ClassLoader';

use Symfony\Component\ClassLoader\UniversalClassLoader;

$loader = new UniversalClassLoader();
$loader->registerNamespace('ProjectA', __DIR__.'/a/src');
$loader->registerNamespace('ProjectB', __DIR__.'/b/src');
$loader->register();

本质上,它的作用是使用spl_register_autoload 注册一个自动加载器,该自动加载器尝试将请求的类与所有注册的路径匹配,如果匹配,则它是必需的。否则自动加载器会继续搜索。

所以你的引导代码所做的就是包含autoload.php,之后所有其他类都将被自动加载。

【讨论】:

谢谢。我提出了另一个想法 - 请参阅上面的编辑/可能的解决方案:)

以上是关于如何使用 __autoload 从多个目录加载类?的主要内容,如果未能解决你的问题,请参考以下文章

_autoload 自动加载类和spl_autoload_register()函数

__autoload() 类文件自动加载函数

php中自动加载类_autoload()和spl_autoload_register()实例详解

spl_autoload_register 和 __autoload()魔术方法

1php----自动加载类 __autoload()函数

本地目录自动加载 - “未找到类”