? ? 在PHP開發(fā)過程中,如果希望從外部引入一個(gè)class,通常會使用include和require方法,去把定義這個(gè)class的文件包含進(jìn)來氛改。這個(gè)在小規(guī)模開發(fā)的時(shí)候,沒什么大問題颜阐。但在大型的開發(fā)項(xiàng)目中平窘,這么做會產(chǎn)生大量的require或者include方法調(diào)用,這樣不因降低效率凳怨,而且使得代碼難以維護(hù)瑰艘,況且require_once的代價(jià)很大。
? ? 在PHP5之前肤舞,各個(gè)php框架如果要實(shí)現(xiàn)類的自動加載紫新,一般都是按照某種約定自己實(shí)現(xiàn)一個(gè)遍歷目錄,自動加載所有符合約定規(guī)則的文件的類或函數(shù)李剖。 當(dāng)然芒率,PHP5之前對面向?qū)ο蟮闹С植⒉皇翘茫惖氖褂靡矝]有現(xiàn)在頻繁篙顺。 在PHP5后偶芍,當(dāng)加載PHP類時(shí),如果類所在文件沒有被包含進(jìn)來德玫,或者類名出錯(cuò)匪蟀,Zend引擎會自動調(diào)用__autoload 函數(shù)。此函數(shù)需要用戶自己實(shí)現(xiàn)__autoload函數(shù)宰僧。 在PHP5.1.2版本后材彪,可以使用spl_autoload_register函數(shù)自定義自動加載處理函數(shù)。當(dāng)沒有調(diào)用此函數(shù)琴儿,默認(rèn)情況下會使用SPL自定義的spl_autoload函數(shù)段化。
1、 __autoload示例:
function ?__autoload($class_name) {
? ? echo ?'__autload?class:',$class_name,'<br />';
}
new ?Demo();
以上的代碼在最后會輸出:__autload class:Demo造成。并在此之后報(bào)錯(cuò)顯示: Fatal error: Class ‘Demo’ not found?
我們一般使用_autoload自動加載類如下:
function ?__autoload($class_name) {
? ? require_once($class_name.?'class.php');
}
$memo=new Demo();
? ? 我們可以看出_autoload至少要做三件事情显熏,第一件事是根據(jù)類名確定類文件名,第二件事是確定類文件所在的磁盤路徑(在我們的例子是最簡單的情況晒屎,類與調(diào)用它們的PHP程序文件在同一個(gè)文件夾下)佃延,第三件事是將類從磁盤文件中加載到系統(tǒng)中现诀。第三步最簡單夷磕,只需要使用include/require即可履肃。要實(shí)現(xiàn)第一步,第二步的功能坐桩,必須在開發(fā)時(shí)約定類名與磁盤文件的映射方法尺棋,只有這樣我們才能根據(jù)類名找到它對應(yīng)的磁盤文件。
? ? 因此绵跷,當(dāng)有大量的類文件要包含的時(shí)候膘螟,我們只要確定相應(yīng)的規(guī)則,然后在__autoload()函數(shù)中碾局,將類名與實(shí)際的磁盤文件對應(yīng)起來荆残,就可以實(shí)現(xiàn)lazy loading的效果。從這里我們也可以看出__autoload()函數(shù)的實(shí)現(xiàn)中最重要的是類名與實(shí)際的磁盤文件映射規(guī)則的實(shí)現(xiàn)净当。
? ? 但現(xiàn)在問題來了内斯,假如在一個(gè)系統(tǒng)的實(shí)現(xiàn)中,假如需要使用很多其它的類庫像啼,這些類庫可能是由不同的開發(fā)工程師開發(fā)俘闯,其類名與實(shí)際的磁盤文件的映射規(guī)則不盡相同。這時(shí)假如要實(shí)現(xiàn)類庫文件的自動加載忽冻,就必須在__autoload()函數(shù)中將所有的映射規(guī)則全部實(shí)現(xiàn)真朗,因此__autoload()函數(shù)有可能會非常復(fù)雜,甚至無法實(shí)現(xiàn)僧诚。最后可能會導(dǎo)致__autoload()函數(shù)十分臃腫遮婶,這時(shí)即便能夠?qū)崿F(xiàn),也會給將來的維護(hù)和系統(tǒng)效率帶來很大的負(fù)面影響湖笨。在這種情況下旗扑,在PHP5引入SPL標(biāo)準(zhǔn)庫,一種新的解決方案,即spl_autoload_register()函數(shù)赶么。
2肩豁、spl_autoload_register()函數(shù)
此函數(shù)的功能就是把函數(shù)注冊至SPL的__autoload函數(shù)棧中,并移除系統(tǒng)默認(rèn)的__autoload()函數(shù)辫呻。下面的例子可以看出:
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(); ?//結(jié)果:SPL?load?class:Test
? ? 語法:bool? spl_autoload_register ( [callback $autoload_function] )? ? 接受兩個(gè)參數(shù):一個(gè)是添加到自動加載棧的函數(shù)清钥,另外一個(gè)是加載器不能找到這個(gè)類時(shí)是否拋出異常的標(biāo)志。第一個(gè)參數(shù)是可選的放闺,并且默認(rèn)指向spl_autoload()函數(shù)祟昭,這個(gè)函數(shù)會自動在路徑中查找具有小寫類名和.php擴(kuò)展或者.ini擴(kuò)展名,或者任何注冊到spl_autoload_extensions()函數(shù)中的其它擴(kuò)展名的文件怖侦。
classCalssLoader{
? ? public static function loader($classname){
? ? ? ? $class_file=strtolower($classname).".php";
? ? ? ?if(file_exists($class_file)){
? ? ? ? ? ? require_once($class_file);
? ? ? ?}
? ? }
}
//?方法為靜態(tài)方法
spl_autoload_register('CalssLoader::loader');
$test = new Test();
一旦調(diào)用spl_autoload_register()函數(shù)篡悟,當(dāng)調(diào)用未定義類時(shí)谜叹,系統(tǒng)會按順序調(diào)用注冊到spl_autoload_register()函數(shù)的所有函數(shù),而不是自動調(diào)用__autoload()函數(shù)搬葬。如果要避免這種情況荷腊,需采用一種更加安全的spl_autoload_register()函數(shù)的初始化調(diào)用方法:
if(false === spl_autoload_functions()){
? ? if(function_exists('__autoload')){
? ? ? ?spl_autoload_registe('__autoload',false);
? ? }
}
spl_autoload_functions()函數(shù)會返回已注冊函數(shù)的一個(gè)數(shù)組,如果SPL自動加載棧還沒有被初始化,它會返回布爾值false。然后急凰,檢查是否有一個(gè)名為__autoload()的函數(shù)存在,如果存在女仰,可以將它注冊為自動加載棧中的第一個(gè)函數(shù),從而保留它的功能抡锈。之后疾忍,可以繼續(xù)注冊自動加載函數(shù)。
還可以調(diào)用spl_autoload_register()函數(shù)以注冊一個(gè)回調(diào)函數(shù),而不是為函數(shù)提供一個(gè)字符串名稱床三。如提供一個(gè)如array('class','method')這樣的數(shù)組,使得可以使用某個(gè)對象的方法一罩。
下一步,通過調(diào)用spl_autoload_call('className')函數(shù)撇簿,可以手動調(diào)用加載器聂渊,而不用嘗試去使用那個(gè)類。這個(gè)函數(shù)可以和函數(shù)class_exists('className',false)組合在一起使用以嘗試去加載一個(gè)類补疑,并且在所有的自動加載器都不能找到那個(gè)類的情況下失敗歧沪。
if(spl_autoload_call('className') &&class_exists('className',false)){
}else{
}
SPL自動加載功能是由spl_autoload() ,spl_autoload_register(), spl_autoload_functions() ,spl_autoload_extensions()和spl_autoload_call()函數(shù)提供的。
《完》
參考:
官方文檔:類的自動加載