PHP的類自動載入機制

來源:互聯網
上載者:User
文章目錄
  •  __autoload樣本:

在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函數。

 

 __autoload樣本:
function __autoload($class_name) {     echo '__autload class:', $class_name, '<br />';  }    new Demo();  

以上的代碼在最後會輸出:__autload class:Demo。並在此之後報錯顯示: Fatal error: Class ‘Demo’ not found

 

我們一般使用_autoload自動載入類如下:

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()函數。

 

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)組合在一起使用以嘗試去載入一個類,並且在所有的自動載入器都不能找到那個類的情況下失敗。

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()函數提供的。

 

原文地址:http://blog.csdn.net/hguisu/article/details/7463333

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.