繁体   English   中英

PHP:AutoLoader 能否在单个 php 文件中加载多个类?

[英]PHP: Is AutoLoader able to load multiple class in a single php file?

引用自自动加载类

许多编写面向对象应用程序的开发人员为每个类定义创建一个 PHP 源文件。 最大的烦恼之一是必须在每个脚本的开头编写一长串所需的包含(每个类一个)。

在 PHP 5 中,这不再是必要的。 spl_autoload_register() 函数注册任意数量的自动加载器,如果当前未定义类和接口,则可以自动加载它们。 通过注册自动加载器,PHP 有最后的机会在它失败并出现错误之前加载类或接口。

那么问题来了,如果一个php文件中有多个类,是否适合autoload使用呢? 还是我必须使用require filepath语句?

比如我在Protobuf\\Client.php下有一个协议文件:

<?php

namespace Protobuf;
class A {
...
}
class B {
...
}

您必须有一些复杂的函数才能从名为Client.php的文件中自动加载这些类。 这个想法是将您的namespace\\classname转换为directory\\filename.php

在这种情况下,您需要将文件A.php然后当您调用new Protobuf\\A()它会找到它。 否则,您将不得不创建一个过于复杂的自动加载器。

假设您确实创建了自动加载器,以便它找到A类,然后您可以在同一个文件中使用B ,但前提是您已经自动加载了A否则您必须进行一些算法才能知道AB位于同一页面上.

我会执行上述模式或Magento 等应用程序采用的模式,通过替换下划线将类名转换为目录路径:

$class = new Core_Classes_MyClass_Client();

您的自动加载器将替换下划线并加载:

Core/Classes/MyClass/Client.php //or similar scheme

这对我来说是一种简单的方法,但我更喜欢使用命名空间和类。 上述方法目前不受欢迎,从命名的角度来看,很容易混淆,因为很多类可能在同一个文件夹中,或者嵌套在子文件夹的深处。 你可以得到一些非常长的类命名。

扩展 Rasclatts 非常翔实的答案,

理想情况下,在自动加载时将类分开总是很好的做法。 我强烈建议研究 Composers PSR-0 Namespace Autoloading

PSR-0 允许您将所有类精美地组织到具有无限深度的子文件夹中,采用以下文件夹结构

\system
 - Members
     - Members.php
 - Auth
     - Auth.php
 - Database
     - Database.php

对于这个例子,在上面的每个 php 文件中,你都会有namespace MyNameSpace; 在您的类声明之前,然后在您的composer.json您将有类似于(文档)的内容:

"autoload": {
    "psr-0": { "MyNameSpace": "/system" }
}

Composer 应该安装在您的本地/主机计算机上,以便您编译自动加载文件、打开终端并导航到您的项目目录并键入:

composer dump-autoload -o

现在一切都井井有条,您可以访问类似于以下内容的类:

\MyNameSpace\Auth::staticFunction();

是和否,但是如果我为一个非常大的 WSDL 生成(自动)类和方法呢?

即。 对于一百多个方法,您可能有一百个methodRequest(作为类对象),接下来的一百个methodResponse(作为类对象)和大型数组,即ClassMap。

有时最好在一个文件中处理这些东西,特别是在没有好的 WSDL 文档的情况下进行开发时。

为了直接回答这个问题,我们可以在composer.json使用classmap for autoload来支持包含多个类的单个文件。

例如,我们将支持单个文件Protobuf\\Client.php ,其中包含两个类AB

<?php

namespace Protobuf;

class A {
}

class B {
}

我们添加classmapcomposer.json如下:

{
    "name": "hailong/myproj",
    "autoload": {
        "classmap": [
            "Protobuf/Client.php"
        ]
    }
}

然后,在main.php我们可以使用A类和B类:

<?php

require __DIR__.'/vendor/autoload.php';

use Protobuf\A;
use Protobuf\B;

$a = new A();

$b = new B();

最后,不要忘记在更改composer.json后运行composer dump-autoload ,这将神奇地生成我们main.php中所需的autoload.php

对于还不熟悉作曲家的人,这里是我们将得到的最终文件结构:

myproj % tree
.
├── Protobuf
│   └── Client.php
├── composer.json
├── main.php
└── vendor
    ├── autoload.php
    └── composer
        ├── ClassLoader.php
        ├── LICENSE
        ├── autoload_classmap.php
        ├── autoload_namespaces.php
        ├── autoload_psr4.php
        ├── autoload_real.php
        ├── autoload_static.php
        └── installed.json

3 directories, 12 files

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM