PHP Conference Japan 2024

spl_autoload_register

(PHP 5 >= 5.1.0, PHP 7, PHP 8)

spl_autoload_register将给定函数注册为 __autoload() 实现

描述

spl_autoload_register(?callable $callback = null, bool $throw = true, bool $prepend = false): bool

将函数注册到 spl 提供的 __autoload 队列中。如果队列尚未激活,则会将其激活。

如果您的代码具有现有的 __autoload() 函数,则必须在 __autoload 队列中显式注册此函数。这是因为 spl_autoload_register() 将有效地替换 __autoload() 函数的引擎缓存,方法是使用 spl_autoload()spl_autoload_call()

如果必须有多个自动加载函数,spl_autoload_register() 允许这样做。它有效地创建了一个自动加载函数队列,并按定义顺序运行每个函数。相比之下,__autoload() 只能定义一次。

参数

callback

正在注册的自动加载函数。如果为 null,则将注册 spl_autoload() 的默认实现。

callback(string $class): void

class 将不包含完全限定标识符的前导反斜杠。

throw

此参数指定 spl_autoload_register()callback 无法注册时是否应抛出异常。

警告

从 PHP 8.0.0 开始,此参数将被忽略,如果设置为 false,则会发出通知。spl_autoload_register() 现在将在无效参数上始终抛出 TypeError

prepend

如果为 truespl_autoload_register() 将在自动加载队列中将自动加载器预置而不是附加它。

返回值

成功时返回 true,失败时返回 false

变更日志

版本 描述
8.0.0 callback 现在可以为 null。

示例

示例 #1 spl_autoload_register() 作为 __autoload() 函数的替代

<?php

// function __autoload($class) {
// include 'classes/' . $class . '.class.php';
// }

function my_autoloader($class) {
include
'classes/' . $class . '.class.php';
}

spl_autoload_register('my_autoloader');

// 或者,使用匿名函数
spl_autoload_register(function ($class) {
include
'classes/' . $class . '.class.php';
});

?>

示例 #2 类未加载的 spl_autoload_register() 示例

<?php

namespace Foobar;

class
Foo {
static public function
test($class) {
print
'[['. $class .']]';
}
}

spl_autoload_register(__NAMESPACE__ .'\Foo::test');

new
InexistentClass;

?>

以上示例将输出类似以下内容

[[Foobar\InexistentClass]]
Fatal error: Class 'Foobar\InexistentClass' not found in ...

示例 #3 标识符将不带前导反斜杠提供

<?php

spl_autoload_register
(static function ($class) {
var_dump($class);
});

class_exists('RelativeName');
class_exists('RelativeName\\WithNamespace');
class_exists('\\AbsoluteName');
class_exists('\\AbsoluteName\\WithNamespace');

?>

以上示例将输出

string(12) "RelativeName"
string(26) "RelativeName\WithNamespace"
string(12) "AbsoluteName"
string(26) "AbsoluteName\WithNamespace"

参见

添加注释

用户贡献的注释 25 条注释

206
a dot schaffhirt at sedna-soft dot de
15 年前
对于使用命名空间类的 PHP 5.3 用户来说,这是一个好消息

当您创建与包含类的命名空间匹配的子文件夹结构时,您甚至永远不必定义自动加载器。

<?php
spl_autoload_extensions
(".php"); // 用逗号分隔的列表
spl_autoload_register();
?>

建议对所有类只使用一个扩展名。PHP(更准确地说,是 spl_autoload)会为您完成其余工作,并且比语义上等效的自定义自动加载函数(如以下函数)更快

<?php
function my_autoload ($pClassName) {
include(
__DIR__ . "/" . $pClassName . ".php");
}
spl_autoload_register("my_autoload");
?>

我使用以下设置对它们进行了比较:有 10 个文件夹,每个文件夹有 10 个子文件夹,每个子文件夹有 10 个子文件夹,每个子文件夹包含 10 个类。

要加载并实例化这 1000 个类(无参数的无操作构造函数),在对每种方法进行 10 次命令行调用的系列中,用户定义的自动加载函数方法平均比 spl_autoload 函数慢 50 毫秒。

我进行此基准测试是为了确保我不会推荐以后可能被称为“不错,但很慢”的东西。

此致,
24
nemanja
7 年前
即使使用了自动加载(SPL),类继承似乎也不起作用。仅仅是 PHP 引擎无法找到父(继承)类。PHP 5.6 和 7.0 在这一点上表现完全相同,这违背了自动加载的初衷。

在我看来,修复它很容易,因为自动加载器能够毫无问题地找到所有第一级类,它只需要在父类上也递归地遵循相同的路径。

<?php
//使用默认的 SPL 自动加载器,命名空间与目录结构一一对应,文件名全部小写。
//这仅适用于第一级类,对于继承来说不起作用,它无法找到父类。
spl_autoload_register();

//如果希望能够自动加载父类,这是丑陋但有效的代码。
spl_autoload_register(function ($class){
require_once
__DIR__ . '/' . strtolower(str_replace('\\', '/', $class) . '.php');
});
30
eyeofmidas at gmail dot com
9 年前
从使用 __autoload() 切换到使用 spl_autoload_register 时,请记住会话的反序列化可能会触发类查找。

这按预期工作
<?php
session_start
();
function
__autoload($class) {
...
}
?>

当使用从会话反序列化的类时,这将导致 "__PHP_Incomplete_Class_Name" 错误。
<?php
session_start
();
function
customAutoloader($class) {
...
}
spl_autoload_register("customAutoloader");
?>

因此,您需要确保在调用 session_start() 之前完成 spl_autoload_register。

正确
<?php
function customAutoloader($class) {
...
}
spl_autoload_register("customAutoloader");
session_start();
?>
43
(delphists) at (apollo) dot (lv)
13 年前
当使用 spl_autoload_register() 与类方法时,它似乎只能使用公共方法,但如果从类内部注册,它也可以使用私有/受保护的方法。
<?php

class ClassAutoloader {
public function
__construct() {
spl_autoload_register(array($this, 'loader'));
}
private function
loader($className) {
echo
'Trying to load ', $className, ' via ', __METHOD__, "()\n";
include
$className . '.php';
}
}

$autoloader = new ClassAutoloader();

$obj = new Class1();
$obj = new Class2();

?>

输出
--------
Trying to load Class1 via ClassAutoloader::loader()
Class1::__construct()
Trying to load Class2 via ClassAutoloader::loader()
Class2::__construct()
15
iam at thatguy dot co dot za
9 年前
<?php

// 使用 SPL_AUTOLOAD_REGISTER 方法从多个目录自动加载类文件示例。
// 它自动加载它找到的任何以 class.<classname>.php(小写)开头的文件,例如:class.from.php、class.db.php
spl_autoload_register(function($class_name) {

// 定义一个目录数组,按其优先级顺序进行迭代。
$dirs = array(
'project/', // 项目特定类(+核心覆盖)
'classes/', // 核心类示例
'tests/', // 单元测试类,如果使用 PHP-Unit
);

// 循环遍历每个目录以加载所有类文件。它只会加载一次文件。
// 如果它稍后在目录中找到相同的类,它将忽略它!因为 require once!
foreach( $dirs as $dir ) {
if (
file_exists($dir.'class.'.strtolower($class_name).'.php')) {
require_once(
$dir.'class.'.strtolower($class_name).'.php');
return;
}
}
});
8
kuzawinski dot marcin at nospam dot gmail dot com
3 年前
从 PHP 8.0 开始,spl_autoload_register() 始终会在无效参数上抛出 TypeError,因此第二个参数 `throw` 将被忽略,如果将其设置为 False,则会发出通知。
19
florent at mediagonale dot com
18 年前
如果您的自动加载函数是类方法,您可以使用指定类和要运行的方法的数组来调用 spl_autoload_register。

* 您可以使用静态方法
<?php

class MyClass {
public static function
autoload($className) {
// ...
}
}

spl_autoload_register(array('MyClass', 'autoload'));
?>

* 或者您可以使用实例
<?php
class MyClass {
public function
autoload($className) {
// ...
}
}

$instance = new MyClass();
spl_autoload_register(array($instance, 'autoload'));
?>
19
anthon at piwik dot org
14 年前
仔细考虑从注册的自动加载器中抛出异常。

如果您注册了多个自动加载器,并且一个(或多个)在后面的自动加载器加载类之前抛出异常,则会抛出堆叠异常(并且必须捕获),即使类已成功加载。
16
a dot schaffhirt at sedna-soft dot de
11 年前
我之前在这里所说的话仅在 Windows 上有效。当您在不带任何参数的情况下调用 spl_autoload_register() 时注册的内置默认自动加载器只是将限定类名加上注册的文件扩展名(.php)添加到每个 include 路径中,并尝试包含该文件。

示例(在 Windows 上)

包含路径
- "."
- "d:/projects/phplib"

要加载的限定类名
network\http\rest\Resource

以下是发生的事情

PHP 尝试加载
'.\\network\\http\\rest\\Resource.php'
-> 文件未找到

PHP 尝试加载
'd:/projects/phplib\\network\\http\\rest\\Resource.php'
-> 文件已找到并包含

请注意文件路径中的斜杠和反斜杠。在 Windows 上,这完美地工作,但在 Linux 机器上,反斜杠不起作用,此外文件名区分大小写。

这就是为什么在 Linux 上快速简便的方法是将这些限定类名转换为斜杠并转换为小写,然后像这样传递给内置自动加载器

<?php
spl_autoload_register
(
function (
$pClassName) {
spl_autoload(strtolower(str_replace("\\", "/", $pClassName)));
}
);
?>

但这意味着您必须将所有类保存为小写文件名。否则,如果您省略 strtolower 调用,则必须使用与文件名完全指定的类名,这对于使用非直观大小写定义的类名(例如 XMLHttpRequest)来说可能会很麻烦。

我更喜欢小写方法,因为它更容易使用,并且可以在部署时自动完成文件名转换。
22
Anonymous
14 年前
在区分大小写的文件系统上使用此函数时要小心。

<?php
spl_autoload_extensions
('.php');
spl_autoload_register();
?>

我在 OS X 上进行开发,一切正常。但是当发布到我的 Linux 服务器时,没有一个类文件被加载。我不得不将所有文件名都转换为小写,因为调用类 "DatabaseObject" 会尝试包含 "databaseobject.php",而不是 "DatabaseObject.php"

我想我会回到使用速度较慢的 __autoload() 函数,这样我就可以保持我的类文件可读性。
2
abhijeet dot sweden at gmail dot com
2 年前
类型#1

<?php

require_once('Second.php');
require_once(
'First.php');

$Second = new Second;
$Second->run_second();

$First = new First;
$First->run_first();

?>

---------------------------------------------------------
类型#2

<?php

spl_autoload_register
(function($class){
//
require_once($class.'.php');
});

$Second = new Second;
$Second->run_second();

$First = new First;
$First->run_first();

?>
6
rayro at gmx dot de
14 年前
在自动加载函数中通过 eval 创建类从来都不是一个好主意,并且是一个不可取的概念。
这些异常应该是一个不错的功能,但我认为任何人都能够在没有这种方法的情况下处理它。目前我还没有意识到这有什么好处......

正如我可能注意到的,class_exists() 将始终定义你只想检查是否存在 类,因此将始终返回 true
<?php
function EvalIsEvil($class) {
eval(
'class '.$className.'{}');
}
spl_autoload_register('EvalIsEvil');
if (
class_exists($s="IsMyModuleHere")) {
// 这不是一个模块,但可以通过 eval() 获取...
return new $s();
}
?>
4
phil at propcom dot co dot uk
11 年前
需要注意的是,如果 E_STRICT 错误触发错误处理程序,而错误处理程序又尝试使用尚未加载的类,则不会调用自动加载程序。

在这种情况下,您应该手动加载错误处理程序所需的类。
3
rnealxp at yahoo dot com
7 年前
我现在使用 spl_autoload_register,并且无法回头。所以让我在这里为你收集我学到的东西......
1.) 这些文档中提到的关于文件名大小写敏感性(Windows 与 Linux/Mac)的问题:只有在你调用 spl_autoload_register 时不提供自己的自定义函数作为参数时,它才会发挥作用。你的函数应该接受一个参数,该参数将是你的代码当前尝试访问的类名。我观察到类名与你在代码库中实际使用的字母大小写相同(混合大小写或不混合大小写)。我没有使用命名空间,但作为最佳实践,为了使你的实现简单直观和可预测,请使用 fileName===className(1:1)。
2.) 我经常在我的代码库的目录结构发生变化时重构它。我没有使用命名空间,但即使我使用了命名空间,我也希望在我的命名空间层次结构和我的目录结构层次结构之间进行解耦。我喜欢我的目录层次结构对于获取我想使用的代码来说很直观。为了避免手动告诉我的自动加载程序每个文件/类的路径,我在静态变量中缓存了几个数组,这些数组在一个数组中存储文件/类名,在另一个数组中存储文件的文件夹路径。在第一次调用我的函数后设置缓存(静态变量)后,只需要在文件名数组中查找即可。为了安全起见,并且为了避免问题,所有目录层次结构(你的命名空间)中的 php 文件名都不应该相同 - 无论如何我更喜欢这种做法,因此,我更喜欢在我的单个命名空间中使用唯一的类名(尽管它还没有明确定义)。我在我的函数中内置了一个检查,以确保所有 php 文件名/类都是唯一的。
3.) 我转换了许多以前在全局空间中有一组函数的文件到*抽象类*,这些类具有私有静态变量和方法,当然还有公共静态方法。因此,这些函数集现在封装在对象中,并且这些对象现在是自动加载的。仅仅为了这个好处,我将永远不会再在全局空间中使用函数,除了我的自动加载函数和其他异常(例如)。
4.) 我的自动加载函数仅使用内置的 php 语言结构和操作,并且没有外部依赖项。
5.) 如果你在代码库中的某个地方使用了函数 class_exists(),请意识到,除非你将第二个参数传递为 false,否则你将触发自动加载程序来加载该类。我当然偶然发现了这一点。我的用例是我不希望加载该类:我只想在使用该类时(在错误处理程序方法中)采取一些措施。
6.) 如果你使用函数 method_exists(),你肯定会触发加载该类(这很有道理,因为你已经决定深入查找特定方法)。
7.) 我在这里认可了别人的想法:我还选择为要加载的类调用一个 init() 方法,如果它存在的话。这使我无需从外部手动调用,更不用说管理如何以及从哪里进行调用。以这种自动化的方式设置你的对象并使其准备好工作非常有用。
8.) 正如其他人所说,我也使用 require() 而不是 require_once(),因为第一个足以生成错误,如果已经加载,则不会调用该函数。
9.) 如果由于某种原因我在缓存的数组中找不到类名,我仍然会明知故犯地调用 require(),传递我未考虑在内的类名,以生成和揭示问题(这当然是不希望发生的!)。
10.) 同样,我确保所有类名都是唯一的。如果我观察到不唯一,我将再次对 require() 进行错误调用,如下所示:require('FoundMultiplesOfClassFile.php'); 以揭示问题。(我还没有,而且你可能也不应该,注册任何复杂的错误处理程序,所以对我来说这和任何其他东西一样好)。
2
daniel at amnistechnology dot com
12 年前
巧妙地 - 并且有用地 - 我注意到(至少在 PHP 5.3 上)这些自动加载程序即使在你调用尚未加载的全静态类的公共静态方法时也会“启动”。
4
sebastian dot krebs at kingcrunch dot de
14 年前
看起来,spl_autoload 在调用每个已注册的加载程序后都会测试类是否存在。因此,如果类存在,它会中断链,并且不会调用其他加载程序

<?php
function a ($c) {
echo
"a\n";
class
Bla {} // 通常 "include 'path/to/file.php';"
}
function
b ($c) {
echo
"b\n";
}
spl_autoload_register('a');
spl_autoload_register('b');

$c = new Bla();
?>
-1
Daniel Klein
2 年前
如果你想能够从任何地方运行你的 PHP 脚本,而无需先切换到该目录,请在运行的第一个文件中使用 chdir(__DIR__)(而不是任何包含的文件)

<?php
spl_autoload_register
();
chdir(__DIR__);

出于
某种原因尝试 使用 其他命名空间类中的命名空间类会失败

e.g. 运行 "php src/main.php" 将失败,但 "cd src && php main.php" 将起作用

src/main.php
<?php
spl_autoload_register
();
//chdir(__DIR__); // 取消注释以使其从任何目录运行
MyNamespace\MyClass::foo();

src/MyNamespace/MyClass.php
<?php
namespace MyNamespace;

class
MyClass {
public static function
foo(): void {
echo
MyOtherClass::BAR; // 即使 MyOtherClass.php 在正确的位置,这也会失败
}
}

src/MyNamespace/MyOtherClass.php
<?php
namespace MyNamespace;

class
MyOtherClass {
public const
BAR = 'baz';
}
0
kakkau at grr dot la
9 年前
关于使用附加参数注册自动加载函数的说明。

./alf.home.php
<?php
/*
* 包含自动加载函数别名 ALF 的类
*/
class ALF {
public function
haaahaaahaaa($class = "ALF", $param = "Melmac") {
echo
"I am ".$class." from ".$param.".\n";
}
}
?>

./kate.melmac.php
<?php
require_once("alf.home.php");
/*
* 通常的做法是获取 ALF
* 并注册一个自动加载函数
*/
$alf = new ALF();
spl_autoload_register(array($alf,'haaahaaahaaa'));
$alf->haaahaaahaaa(); // ALF 来自麦尔马克 :)
/*
* 现在让我们尝试自动加载一个类
*/
@$kate = new Kate(); // 这会抛出一个致命错误,因为
// Kate 不来自麦尔马克 :)
?>
我是来自麦尔马克的 ALF。
我是来自麦尔马克的 Kate。

./kate.earth.php
<?php
require_once("alf.home.php");
/*
* 但是如果我们想纠正 Kate 的起源呢?
* 如何在注册时将参数传递给自动加载函数?
*
* spl_autoload_register 不适合这种情况
* 但我们可以尝试在注册期间定义一个可调用对象
*/
spl_autoload_register(function($class){
call_user_func(array(new ALF(),'haaahaaahaaa'), $class, "Earth"); });
/*
* 现在让我们再次尝试自动加载一个类
* Kate 仍然找不到,但我们纠正了她的起源 :)
*/
@$kate = new Kate(); // Kate 来自地球 :)
/*
* 注意:你不能使用上面的注册方式传递在可调用上下文之外创建的 $this 或其他对象。因此,你应该像在开头使用 ALF 一样,将你的自动加载函数替换为一个单独的类。
*
* 注意:你可能无法直接注销你的自动加载函数,因为实例是在另一个上下文中创建的
*/
?>
我是来自地球的 Kate。
0
stanlemon at mac dot com
17 年前
编者注:请求此函数模拟的行为的相应 PHP bug 是 http://bugs.php.net/bug.php?id=42823 。如果在自动加载堆栈中注册了 array($obj, 'nonStaticMethod'),则此函数不起作用——虽然自动加载将被删除,但它将被错误地重新注册。

spl_autoload_register() 方法按 spl_autoload_register() 调用的顺序在其堆栈中注册函数,并且随后如果你希望一个自动加载函数覆盖之前的自动加载函数,则需要注销之前的函数或更改自动加载堆栈的顺序。

例如,假设在你对自动加载函数的默认实现中,如果找不到类,则抛出异常,或者可能抛出致命错误。稍后在你的代码中,你添加了自动加载函数的第二个实现,它将加载先前方法将失败的库。这不会首先调用第二个自动加载器方法,而是会继续在第一个方法上出错。

如前所述,你可以注销出错的现有自动加载器,或者可以创建注销和重新注册自动加载器的机制,以你想要的顺序。

以下是如何考虑重新注册自动加载器的一个示例,以便最新的自动加载器首先被调用,最旧的最后被调用

<?php

// 编者注:对函数添加了一些小的错误和兼容性修复
//

function spl_autoload_preregister( $autoload ) {
// 堆栈中当前没有函数。
if ( ($funcs = spl_autoload_functions()) === false ) {
spl_autoload_register($autoload);
} else {
// 注销现有的自动加载器...
$compat =
version_compare(PHP_VERSION, '5.1.2', '<=') &&
version_compare(PHP_VERSION, '5.1.0', '>=');
foreach (
$funcs as $func) {
if (
is_array($func)) {
// :TRICKY: 存在一些兼容性问题,以及一些
// 需要出错的地方
$reflector = new ReflectionMethod($func[0], $func[1]);
if (!
$reflector->isStatic()) {
throw new
Exception('
此函数与非静态对象方法不兼容
由于 PHP Bug #44144。
'
);
}
// 令人惊讶的是,spl_autoload_register 支持
// Class::staticMethod 回调格式,尽管 call_user_func 没有
if ($compat) $func = implode('::', $func);
}
spl_autoload_unregister($func);
}

// 注册新的,从而将其放在堆栈的前面...
spl_autoload_register($autoload);

// 现在,返回并重新注册我们所有旧的。
foreach ($funcs as $func) {
spl_autoload_register($func);
}
}
}

?>

注意:我还没有测试过它的开销,因此我不确定上面示例的性能影响是什么。
-1
harvey dot NO_SPAM dot robin at gmail dot com
17 年前
此函数足够智能,不会重复添加相同的加载器。这似乎适用于所有不同的加载器格式。示例

<?php
class ALoader
{
static function
load($class) { return true; }
}

function
anotherLoader($class) {
return
true;
}

$F = new ALoader;

spl_autoload_register(array('ALoader', 'load'));
spl_autoload_register(array('ALoader', 'load'));
spl_autoload_register(array($F, 'load'));
spl_autoload_register('anotherLoader');
spl_autoload_register('anotherLoader');
var_dump(spl_autoload_functions());

/*
* PHP5.2 CLI,linux 上的结果。
* array(2) {
* [0]=>
* array(2) {
* [0]=>
* string(7) "ALoader"
* [1]=>
* string(4) "load"
* }
* [1]=>
* string(13) "anotherLoader"
* }
*/
?>
-1
n0mAd at example dot com
6 年前
如果在使用命名空间时需要注册函数,请使用 __NAMESPACE__ 常量来定义名称。

<?php

namespace Foobar;

spl_autoload_register('MyFunction'); // 不正确
spl_autoload_register('\MyFunction');// 不正确
spl_autoload_register(__NAMESPACE__ . '\MyFunction'); // 正确

?>
-2
hajo-p
10 年前
如果你有一个像 "/abc/def/ghi" 这样的目录结构,你的 index.php 位于顶级目录中,但你希望使用以 "def" 或 "ghi" 开头的命名空间

你可以使用例如 set_include_path(__DIR__ . '/abc') 切换 php 的命名空间根目录,然后使用简单的 spl_autoload_register() 函数定义和使用你的命名空间,而无需提供任何参数。

请记住,php 处理程序 "cli" 和 "cli-server" 是特殊情况。
-3
Kurd the Great
11 年前
if(!defined('BASE_PATH')) {
define('BASE_PATH', dirname(__FILE__) . '/');
require BASE_PATH . 'Autoloader.php';
Autoloader::Register();
}

class Autoloader
{
public static function Register() {
return spl_autoload_register(array('Autoloader', 'Load'));
}

public static function Load($strObjectName) {
if(class_exists($strObjectName) === false) {
return false;
}

$strObjectFilePath = BASE_PATH . $strObjectName . '.php';

if((file_exists($strObjectFilePath) === false) || (is_readable($strObjectFilePath) === false)) {
return false;
}

require($strObjectFilePath);
}
}
-2
joneschrisan at aol dot com
8 年前
看起来在 Debian php 的最新更新中,传递 no params 给 spl_autoload 在 linux 上不再起作用了。

它无法将命名空间中的 \'s 替换为文件路径中的 /'s。
-3
neolium at gmail dot com
9 年前
如果你将每个类放在不同的文件中,此自动加载将找到你调用的每个类。

它从你在 $root 变量中指定的根目录开始递归遍历每个目录。

你可以在 $dir_to_not_look_in 数组中指定不想遍历的文件夹(例如,在 MVC 项目中,你不会在 'view' 文件夹中找到任何类);

spl_autoload_register(function($class) {

$root = 'my/root/path';
$file = $class . '.php';
$dir_to_not_look_in = array($directories, $to, $not, $look, $in);

if(!function_exists('load')) {
function load($dir, $file) {
if(file_exists($dir . '/' . $file)) {
require_once $dir . '/' . $file;
} else {
foreach(scandir($dir) as $value) {
if(is_dir($dir. '/' . $value) && !in_array($value, $dir_to_no_look_in))
load($dir. '/' . $value, $file);
}
}
};
}

load($root, $file);

});
To Top