什么叫Autoload?
在某个php执行上下文中,在new某个类,或者静态调用时如果某个类没有找到,php默认会首先触发__autoload回调,由回调尝试去加载类代码文件。这个回调由用户自己实现,通过用户规定的类名到代码文件的映射规则得到代码文件路径,并使用
require/include
函数去加载代码文件。
比如:
<?php
__autoload($clsname) {
require $clsname.'.php';
}
$a = new A();
但是,
__autoload
有一个不好的地方是只允许注册一个回调。同时,因为我们在使用一些第三方类库的时候,经常需要维护各自的autoload
调用规则。所以,这里在php5.1.2
之后,我们大多使用spl_autoload_register
来替代__autoload
。
- spl_autoload_register的用法很简单:
spl_autoload_register(function($clsname){
$clspath = explode('\\',$clsname);
if($clspath[0] === 'web') {
$clspath[0] = 'src';
}
require dirname(__DIR__).DIRECTORY_SEPARATOR.implode(DIRECTORY_SEPARATOR,$clspath).'.php';
});
它的参数其实就是一个__autoload回调,你可以多次调用spl_autoload_register来往队列注册多个回调。
这样,使用autoload函数的好处是,如果我一个代码文件中需要使用100个类,我不需要一个个的将其require进来,我只需要将其按照一定的规范组织代码文件。然后注册一个autoload函数,按照我们的规范来自动的根据类名来找到对应的类文件,并require到当前执行环境。
关于PHP的Autoload,我们不得不提的是PSR0-PSR4规范。这两个规范不是PHP的语言标准的一部分,只是PHP使用自动加载的代码组织过程中的一个标准规范,当然你可以完全不遵循这个规范,但是建议你最好能够遵循。
- 一个简单的AutoLoader实例
/**
* 自动加载器,遵循psr-4规范
* @author fangl
*
*/
class Autoloader {
static $_namespaces = [
'web' => 'src',
];
/**
* 增加命名空间到路径的映射(以帮助自动加载器能够找到对应的路径)
* 注意对应的代码里面的命名空间要和声明一致,否则即使文件正确引入,也会报找不到类文件错误
* @param string $namespace 命名空间(只接受一个字符串)
* @param string $path 命名空间对应的路径
*/
static function addNameSpace($namespace,$path) {
self::$_namespaces[trim($namespace,'\\/')] = trim($path,'\\/');
}
/**
* 获取命名空间的加载路径,如果命名空间不存在,返回原值
* @param string $namespace
* @return Ambigous <unknown, multitype:string , string>
*/
static function getPath($namespace) {
return isset(self::$_namespaces[$namespace])?self::$_namespaces[$namespace]:$namespace;
}
/**
* 自动加载回调函数
* @param string $clsname
*/
static function autoload($clsname) {
$clsname = trim($clsname,'\\/');
$clspath = explode('\\',$clsname);
$clspath[0] = self::getPath($clspath[0]);
require APP_ROOT.DIRECTORY_SEPARATOR.implode(DIRECTORY_SEPARATOR,$clspath).'.php';
}
}
//注册自动加载
spl_autoload_register(__NAMESPACE__.'\Autoloader::autoload');