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函數。
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_register()函数,当调用未定义类时,系统会按顺序调用注册到spl_autoload_register()函数的所有函数,而不是自动调用autoload()函数。如果要避免这种情况,需采用一种更加安全的spl_autoload_register()函数的初始化调用方法:
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 { }
SPL自动加载功能是由spl_autoload() ,spl_autoload_register(), spl_autoload_functions() ,spl_autoload_extensions()和spl_autoload_call()函数提供的。
以上是php類別自動載入機制實例詳解的詳細內容。更多資訊請關注PHP中文網其他相關文章!