PHP: AutoLoader 可以在一个php文件中加载多个class吗?
PHP: Is AutoLoader able to load multiple class in a single php file?
Many developers writing object-oriented applications create one PHP
source file per class definition. One of the biggest annoyances is
having to write a long list of needed includes at the beginning of
each script (one for each class).
In PHP 5, this is no longer necessary. The spl_autoload_register()
function registers any number of autoloaders, enabling for classes and
interfaces to be automatically loaded if they are currently not
defined. By registering autoloaders, PHP is given a last chance to
load the class or interface before it fails with an error.
问题来了,如果一个php文件里有多个类,适合autoload使用吗?还是我必须使用 require filepath
语句?
比如我在Protobuf\Client下有一个协议文件.php:
<?php
namespace Protobuf;
class A {
...
}
class B {
...
}
您需要一些复杂的函数才能从名为 Client.php
的文件中自动加载那些 classes。这个想法是将您的 namespace\classname
翻译成 directory\filename.php
在这种情况下,您需要将文件命名为 A.php
,然后当您调用 new Protobuf\A()
时它会找到它。否则你将不得不创建一个过于复杂的自动加载器。
假设您确实创建了自动加载器,因此它找到了 A
class,那么您可以在同一个文件上使用 B
,但前提是您已经自动加载了 A
否则你必须做一些算法才能知道 A
和 B
在同一页上。
我会采用上述模式或像 Magento 这样的应用程序采用的模式,通过替换下划线将 class 名称转换为目录路径:
$class = new Core_Classes_MyClass_Client();
您的自动加载器将替换下划线并将加载:
Core/Classes/MyClass/Client.php //or similar scheme
这对我来说是一种简单的方法,但我更喜欢使用命名空间和 class。上面的方法目前并不受欢迎,从命名的角度来看,很容易混淆,因为很多 classes 可能在同一个文件夹中或嵌套在子文件夹中。您可以为 classes.
获得一些非常长的命名
为了扩展 Rasclatts 非常翔实的答案,
理想情况下,在自动加载时将 class 分开始终是一个好习惯。我强烈建议查看作曲家 PSR-0 命名空间自动加载
PSR-0 允许您将所有 class 精美地组织到无限深度的子文件夹中,采用以下文件夹结构
\system
- Members
- Members.php
- Auth
- Auth.php
- Database
- Database.php
对于这个例子,在上面的每个 php 文件中,你的 class 声明之前会有 namespace MyNameSpace;
,然后在你的 composer.json
中你会有类似的东西至 (documentation):
"autoload": {
"psr-0": { "MyNameSpace": "/system" }
}
Composer 应该安装在您的 local/host 计算机上以便您编译自动加载文件,打开终端并导航到您的项目目录并键入:
composer dump-autoload -o
现在一切都井井有条,您可以访问您的 classes,类似于:
\MyNameSpace\Auth::staticFunction();
是也不是,
但是,如果我确实为非常大的 WSDL 生成(自动)classes 和方法怎么办?
即。对于超过一百个方法,您可能会有一百个 methodRequests(作为 class 对象),接下来的一百个 methodResponse(作为 Class 对象)和大型数组,即 ClassMap。
有时最好在一个文件中处理这些内容,尤其是在没有良好的 WSDL 文档的情况下进行开发时。
直接回答这个问题,我们可以用classmap
for autoload
in composer.json
来支持单个文件包含多个class.
例如我们将支持单个文件 Protobuf\Client.php
,其中有两个 class A
和 B
:
<?php
namespace Protobuf;
class A {
}
class B {
}
我们在composer.json
中添加classmap
如下:
{
"name": "hailong/myproj",
"autoload": {
"classmap": [
"Protobuf/Client.php"
]
}
}
然后,在main.php
中我们可以使用classA
和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
.
对于还不熟悉 composer 的人,这是我们将得到的最终文件结构:
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
Many developers writing object-oriented applications create one PHP source file per class definition. One of the biggest annoyances is having to write a long list of needed includes at the beginning of each script (one for each class).
In PHP 5, this is no longer necessary. The spl_autoload_register() function registers any number of autoloaders, enabling for classes and interfaces to be automatically loaded if they are currently not defined. By registering autoloaders, PHP is given a last chance to load the class or interface before it fails with an error.
问题来了,如果一个php文件里有多个类,适合autoload使用吗?还是我必须使用 require filepath
语句?
比如我在Protobuf\Client下有一个协议文件.php:
<?php
namespace Protobuf;
class A {
...
}
class B {
...
}
您需要一些复杂的函数才能从名为 Client.php
的文件中自动加载那些 classes。这个想法是将您的 namespace\classname
翻译成 directory\filename.php
在这种情况下,您需要将文件命名为 A.php
,然后当您调用 new Protobuf\A()
时它会找到它。否则你将不得不创建一个过于复杂的自动加载器。
假设您确实创建了自动加载器,因此它找到了 A
class,那么您可以在同一个文件上使用 B
,但前提是您已经自动加载了 A
否则你必须做一些算法才能知道 A
和 B
在同一页上。
我会采用上述模式或像 Magento 这样的应用程序采用的模式,通过替换下划线将 class 名称转换为目录路径:
$class = new Core_Classes_MyClass_Client();
您的自动加载器将替换下划线并将加载:
Core/Classes/MyClass/Client.php //or similar scheme
这对我来说是一种简单的方法,但我更喜欢使用命名空间和 class。上面的方法目前并不受欢迎,从命名的角度来看,很容易混淆,因为很多 classes 可能在同一个文件夹中或嵌套在子文件夹中。您可以为 classes.
获得一些非常长的命名为了扩展 Rasclatts 非常翔实的答案,
理想情况下,在自动加载时将 class 分开始终是一个好习惯。我强烈建议查看作曲家 PSR-0 命名空间自动加载
PSR-0 允许您将所有 class 精美地组织到无限深度的子文件夹中,采用以下文件夹结构
\system
- Members
- Members.php
- Auth
- Auth.php
- Database
- Database.php
对于这个例子,在上面的每个 php 文件中,你的 class 声明之前会有 namespace MyNameSpace;
,然后在你的 composer.json
中你会有类似的东西至 (documentation):
"autoload": {
"psr-0": { "MyNameSpace": "/system" }
}
Composer 应该安装在您的 local/host 计算机上以便您编译自动加载文件,打开终端并导航到您的项目目录并键入:
composer dump-autoload -o
现在一切都井井有条,您可以访问您的 classes,类似于:
\MyNameSpace\Auth::staticFunction();
是也不是, 但是,如果我确实为非常大的 WSDL 生成(自动)classes 和方法怎么办?
即。对于超过一百个方法,您可能会有一百个 methodRequests(作为 class 对象),接下来的一百个 methodResponse(作为 Class 对象)和大型数组,即 ClassMap。
有时最好在一个文件中处理这些内容,尤其是在没有良好的 WSDL 文档的情况下进行开发时。
直接回答这个问题,我们可以用classmap
for autoload
in composer.json
来支持单个文件包含多个class.
例如我们将支持单个文件 Protobuf\Client.php
,其中有两个 class A
和 B
:
<?php
namespace Protobuf;
class A {
}
class B {
}
我们在composer.json
中添加classmap
如下:
{
"name": "hailong/myproj",
"autoload": {
"classmap": [
"Protobuf/Client.php"
]
}
}
然后,在main.php
中我们可以使用classA
和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
.
对于还不熟悉 composer 的人,这是我们将得到的最终文件结构:
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