#1. 为何须要类自动加载机制php
##缘由: 在PHP开发过程当中,若是但愿从外部引入一个class,一般会使用include和require方法,去把定义这个class的文件包含进来。这个在小规模开发的时候,没什么大问题。但在大型的开发项目中,这么作会产生大量的require或者include方法调用,这样不因下降效率,并且使得代码难以维护,何况require_once的代价很大。数组
##解决 在PHP5以前,各个PHP框架若是要实现类的自动加载,通常都是按照某种约定本身实现一个遍历目录,自动加载全部符合约定规则的文件的类或函数。 固然,PHP5以前对面向对象的支持并非太好,类的使用也没有如今频繁。安全
在PHP5后,当加载PHP类时,若是类所在文件没有被包含进来,或者类名出错,Zend引擎会自动调用__autoload 函数。此函数须要用户本身实现__autoload函数。框架
在PHP5.1.2版本后,可使用spl_autoload_register函数自定义自动加载处理函数。当没有调用此函数,默认状况下会使用SPL自定义的spl_autoload函数。函数
教程ui
#2. __autoload(类名)函数.net
一、 __autoload示例: function __autoload($class_name) { echo '__autload class:', $class_name, '<br />'; } new Demo();
以上的代码在最后会输出:__autload class:Demo
。code
并在此以后报错显示: Fatal error: Class ‘Demo’ not found
对象
咱们通常使用_autoload自动加载类以下:blog
function __autoload($class_name) { require_once ($class_name . “class.php”); } $memo= new Demo();
##1._autoload三件事 咱们能够看出_autoload至少要作三件事情, 第一件事是根据类名肯定类文件名,第二件事是肯定类文件所在的磁盘路径(在咱们的例子是最简单的状况,类与调用它们的PHP程序文件在同一个文件夹下),第三件事是将类从磁盘文件中加载到系统中。第三步最简单,只须要使用include/require便可。要实现第一步,第二步的功能,必须在开发时约定类名与磁盘文件的映射方法,只有这样咱们才能根据类名找到它对应的磁盘文件。
所以,当有大量的类文件要包含的时候,咱们只要肯定相应的规则,而后在__autoload()函数中,将类名与实际的磁盘文件对应起来,就能够实现lazy loading的效果。
从这里咱们也能够看出__autoload()函数的实现中最重要的是类名与实际的磁盘文件映射规则的实现。
##2._autoload的问题
但如今问题来了,假如在一个系统的实现中,假如须要使用不少其它的类库,这些类库多是由不一样的开发工程师开发,其类名与实际的磁盘文件的映射规则
不尽相同。这时假如要实现类库文件的自动加载,就必须在__autoload()函数中将全部的映射规则所有实现,所以__autoload()函数有可能会很是复杂,甚至没法实现。
最后可能会致使__autoload()函数十分臃肿,这时即使可以实现,也会给未来的维护和系统效率带来很大的负面影响。在这种状况下,在PHP5引入SPL标准库,一种新的解决方案,即spl_autoload_register()函数。
##3._autoload废弃的缘由
咱们一个php项目依赖于多个框架,若是每个框都都有一个_autoload方法,那他就会报函数重复定义的错误,
#3. spl_autoload_register()函数。 ##php版本>5.3 spl_autoload_register()函数取代._autoload函数
此函数的功能就是把函数注册至SPL的__autoload函数栈中,并移除系统默认的__autoload()函数。下面的例子能够看出:
function __autoload($class_name) { echo '__autload class:', $class_name, '<br />'; } function classLoader($class_name) { echo 'SPL load class:', $class_name, '<br />'; } spl_autoload_register('classLoader'); new Test();//结果:SPL load class:Test
语法:bool spl_autoload_register ( [callback $autoload_function] ) 接受两个参数:一个是添加到自动加载栈的函数,另一个是加载器不能找到这个类时是否抛出异常的标志。第一个参数是可选的,而且默认指向spl_autoload()函数,这个函数会自动在路径中查找具备小写类名和.php扩展或者.ini扩展名,或者任何注册到spl_autoload_extensions()函数中的其它扩展名的文件。
##注册类方法
<?php class CalssLoader { public static function loader($classname) { $class_file = strtolower($classname).".php"; if (file_exists($class_file)){ require_once($class_file); } } } // 方法为静态方法 spl_autoload_register('CalssLoader::loader'); $test = new Test();
一旦调用spl_autoload_register()函数,当调用未定义类时,系统会按顺序调用注册到spl_autoload_register()函数的全部函数,而不是自动调用__autoload()函数。若是要避免这种状况,需采用一种更加安全的spl_autoload_register()函数的初始化调用方法:
[php] view plain copy print? if(false === spl_autoload_functions()){ if(function_exists('__autoload')){ spl_autoload_registe('__autoload',false); } }
spl_autoload_functions()函数会返回已注册函数的一个数组,若是SPL自动加载栈尚未被初始化,它会返回布尔值false。而后,检查是否有一个名为__autoload()的函数存在,若是存在,能够将它注册为自动加载栈中的第一个函数,从而保留它的功能。以后,能够继续注册自动加载函数。
还能够调用spl_autoload_register()函数以注册一个回调函数,而不是为函数提供一个字符串名称。如提供一个如array('class','method')这样的数组,使得可使用某个对象的方法。
下一步,经过调用spl_autoload_call('className')函数,能够手动调用加载器,而不用尝试去使用那个类。这个函数能够和函数class_exists('className',false)组合在一块儿使用以尝试去加载一个类,而且在全部的自动加载器都不能找到那个类的状况下失败。
f(spl_autoload_call('className') && class_exists('className',false)){ } else { }
#4. SPL自动加载功能
SPL自动加载功能是由 spl_autoload() , spl_autoload_register(), spl_autoload_functions() ,spl_autoload_extensions()和spl_autoload_call()函数提供的。
#5. 注意点: 注册的是函数名不要小括号
<?php function __autoload($class_name) { echo '__autload class:', $class_name, '<br />'; } spl_autoload_register('autoload1'); spl_autoload_register('autoload2'); function autoload1($class){ require __DIR__.''.$class.'.php'; } function autoload2($class){ require __DIR__.''.$class.'.php'; } ?>