在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函數。
1、 __autoload示例:
function __autoload($class_name) { echo '__autload class:', $class_name, '<br />'; } new Demo();
以上的代碼在最后會輸出:__autload class:Demo。
并在此之后報錯顯示: Fatal error: Class ‘Demo’ not found
我們一般使用_autoload自動加載類如下:
<?php function __autoload($class_name) { require_once ($class_name . “class.php”); } $memo= new Demo();
我們可以看出_autoload至少要做三件事情,第一件事是根據類名確定類文件名,第二件事是確定類文件所在的磁盤路徑(在我們的例子是最簡單的情況,類與調用它們的PHP程序文件在同一個文件夾下),第三件事是將類從磁盤文件中加載到系統中。第三步最簡單,只需要使用include/require即可。要實現第一步,第二步的功能,必須在開發時約定類名與磁盤文件的映射方法,只有這樣我們才能根據類名找到它對應的磁盤文件。
因此,當有大量的類文件要包含的時候,我們只要確定相應的規則,然后在__autoload()函數中,將類名與實際的磁盤文件對應起來,就可以實現lazy loading的效果。從這里我們也可以看出__autoload()函數的實現中最重要的是類名與實際的磁盤文件映射規則的實現。
但現在問題來了,假如在一個系統的實現中,假如需要使用很多其它的類庫,這些類庫可能是由不同的開發工程師開發,其類名與實際的磁盤文件的映射規則不盡相同。這時假如要實現類庫文件的自動加載,就必須在__autoload()函數中將所有的映射規則全部實現,因此__autoload()函數有可能會非常復雜,甚至無法實現。最后可能會導致__autoload()函數十分臃腫,這時即便能夠實現,也會給將來的維護和系統效率帶來很大的負面影響。在這種情況下,在PHP5引入SPL標準庫,一種新的解決方案,即spl_autoload_register()函數。
2、spl_autoload_register()函數
此函數的功能就是把函數注冊至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_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 { }
SPL自動加載功能是由spl_autoload() ,spl_autoload_register(), spl_autoload_functions() ,spl_autoload_extensions()和spl_autoload_call()函數提供的。
轉自:http://blog.csdn.net/hguisu/article/details/7463333
原創文章,作者:s19930811,如若轉載,請注明出處:http://www.www58058.com/2980