https://www.jb51.net/article/134372.htm
问题
传统上,在PHP里,当我们要用到一个class文件的时候,我们都得在文档头部require或者include一下:
<?php
require_once('../includes/functions.php');
require_once('../includes/database.php');
require_once('../includes/user.php');
...
但是一旦要调用的文档多了,就得每次都写一行,瞅着也不美观,有什么办法能让PHP文档自动加载呢?
方法:使用__autoload
<?php
function __autoload($class_name)
{
require "./{$class_name}.php";
}
对,可以使用PHP的魔法函数__autoload(),上面的示例就是自动加载当前目录下的PHP文件。
当然,实际当中,我们更可能会这么来使用:
<?php
function __autoload($class_name)
{
$name = strtolower($class_name);
$path = "../includes/{$name}.php";
if(file_exists($path)){
require_once($path);
}else{
die("the file {$class_name} could not be found");
}
}
__autoload存在的问题:做不到多个开发人员使用不同的自定义的autoloader,
除非大家都提前说好了,都使用一个__autoload,涉及到改动了就进行版本同步,这很麻烦。
__autoload函数马上要在7.2版本的PHP中弃用了。
取而代之的是一个叫spl_autoload_register()的东东,它的好处是可以自定义多个autoloader.
方法二:spl_autoload_register
如果需要多条 autoload 函数,spl_autoload_register() 满足了此类需求。
它实际上创建了 autoload 函数的队列,按定义时的顺序逐个执行。
相比之下, __autoload() 只可以定义一次。
autoload_function
欲注册的自动装载函数。如果没有提供任何参数,则自动注册 autoload 的默认实现函数spl_autoload()。
throw
此参数设置了 autoload_function 无法成功注册时, spl_autoload_register()是否抛出异常。
prepend
如果是 true,spl_autoload_register() 会添加函数到队列之首,而不是队列尾部。
//使用一个全局函数
function Custom()
{
require_once('...');
}
spl_autoload_register('Custom');
//使用一个class当中的static方法
class MyCustomAutoloader
{
static public function myLoader($class_name)
{
require_once('...');
}
}
//传array进来,第一个是class名,第二个是方法名
spl_autoload_register(['MyCustomAutoloader','myLoader']);
//甚至也可以用在实例化的object上
class MyCustomAutoloader
{
public function myLoader($class_name)
{
}
}
$object = new MyCustomAutoloader;
spl_autoload_register([$object,'myLoader']);
使用autoload,无论是__autoload(),还是spl_autoload_register(),相比于require或include,
好处就是autoload机制是lazy loading,也即是并不是你一运行就给你调用所有的那些文件,而是只有你
用到了哪个,比如说new了哪个文件以后,才会通过autoload机制去加载相应文件。
总结
__autoload只可以定义一次,加载引用的函数直接写在__autoload(){}里面
spl_autoload_register可以定义多次,并且可以引用自定义的加载引用类的函数,相当与有一个加载函数的队列,而上面的函数只允许一个,不够灵活强大!