背景
在php5之前,如果要使用某个类,需要用include/require将其包含进来即可。随着项目规模不断变大,这种方式会造成一些问题:需要大量的reuqure/include语句;加载不必要的类;维护起来特别困难;一个引用错误整体会报错。
php5为了解决这个问题,提供了“类的自动加载"机制。
autoload机制
如果加载一个类,该类没有被加载,则自动运行__autoload函数
function __autoload($classname) {
require_once ($classname . "class.php");
}
该函数实现步骤:
第一步:将类名转化成文件名
第二补:确定实际的磁盘地址
第三部:require/include
所以,我们只需要在__autoload()函数中,将类名和文件名按照一定规则对应起来,就可以实现类的自动加载。
问题出现了,在实际开发中,对应规则并不是一成不变的,如果类和实际磁盘文件的映射规则有很多,是不是__autoload()函数就会显得十分臃肿?
继续探究php的自动加载机制,发现并不是简单的调用__autoload()函数这么简单。其实是这个样子:
检查执行器全局变量函数指针autoload_func如果为null
定义了__autoload()函数,执行,并返回结果。
未定义__autoload()函数,报告错误并退出
检查执行器全局变量函数指针autoload_func不为null,执行autoload_func函数。autoload_func函数是干啥的呢,继续往下看。
SPL autoload机制
SPL是Standard PHP Library(标准PHP库)的缩写,spl可以将函数指针autoload_func指向自己实现的自动加载函数。spl有两个函数:sql_autoload和sql_autoload_call。
spl_autoload函数,有两个参数,一个是类名,一个是文件拓展名,可以多个。实际上和__aotoload差不多。
spl_autoload_call内部有一个全局变量autoload_functions,本质上就是一个链表,一个元素指向一个指针,指向一个自动加载类的函数,在执行的时候按照顺序一个一个自动加载方法去加载,找到类就终止。
php自动加载到底怎么走呢?有一个函数起到了至关重要的作用,spl_autoload_register。
spl_autoload_register()没有参数就是让autoload_func执行spl_autoload。有参数,比如定义了一个自动加载类方法test,spl_autoload_register(test())就是让autoload_func执行spl_autoload_call