先給大家說下什么是命名空間。
“什么是命名空間?從廣義上來說,命名空間是一種封裝事物的方法。在很多地方都可以見到這種抽象概念。例如,在操作系統中目錄用來將相關文件分組,對于目錄中的文件來說,它就扮演了命名空間的角色。具體舉個例子,文件 foo.txt 可以同時在目錄/home/greg 和 /home/other 中存在,但在同一個目錄中不能存在兩個 foo.txt 文件。另外,在目錄 /home/greg 外訪問 foo.txt 文件時,我們必須將目錄名以及目錄分隔符放在文件名之前得到 /home/greg/foo.txt。這個原理應用到程序設計領域就是命名空間的概念。”
PHP的自動加載就是我們加載實例化類的時候,不需要手動去寫require來導入這個class.php文件,程序自動幫我們加載導入進來。配合命名空間規范,我們可以在復雜系統中很輕松的處理不同類的加載和調用問題。
1. 自動加載的原理以及__autoload的使用
自動加載的原理,就是在我們實例化一個 class 的時候,PHP如果找不到這個類,就會去自動調用本文件中的 __autoload($class_name) 方法,我們new的這個class_name 就成為這個方法的參數。所以我們就可以在這個方法中根據我們需要new class_name的各種判斷和劃分就去require對應的路徑類文件,從而實現自動加載。
我們先來看下 __autoload() 的自動調用,舉個栗子:
index.php
$db = new Db();
如果我們不手動導入Db類,程序可能會報錯,說找不到這個類:
Fatal error: Uncaught Error: Class 'DB' not found in D:webhellowebademo2017autoloadindex.php:2 Stack trace: #0 {main} thrown in D:webhellowebademo2017autoloadindex.php on line 2
那么,我們現在加入 __autoload() 這個方法再看看:
- $db = new DB();
- function __autoload($className) {
- echo $className;
- exit();
- }
根據上面自動加載機制的描述,會輸出:Db, 也就是我們需要new 的類的類名。所以,這個時候我們就可以在 __autoload() 方法里,根據需要去加載類庫文件了。
2. spl_autoload_register自動加載
如果是小項目,用 __autoload() 就能實現基本的自動加載了。但是如果一個項目很大,或者需要不同的自動加載來加載不同路徑的文件,這個時候__autoload就杯具了,因為一個項目中只允許有一個 __autoload() 函數,因為 PHP 不允許函數重名了,也就是說你不能聲明2個 __autoload() 函數文件,否則會報致命錯誤。那怎么辦呢?放心,你想到的,PHP大神早已經想到。 所以 spl_autoload_register() 這樣又一個牛逼函數誕生了,并且取而代之它。它執行效率更高,更靈活。
先看下它如何使用,在index.php中加入以下代碼。
- spl_autoload_register(function($className){
- if (is_file('./Lib/' . $className . '.php')) {
- require './Lib/' . $className . '.php';
- }
- });
- $db = new Db();
- $db::test();
在LibDb.php文件中加入以下代碼:
- class Db
- {
- public static function test()
- {
- echo 'Test';
- }
- }
運行index.php后,當調用 new Db() 時, spl_autoload_register 會自動去lib/目錄下查找對應的Db.php文件,成功后并且能夠執行 $db::test(); 。同樣如果在Lib目錄下有多個php類文件,都可以在index.php中直接調用,而不需要使用 require 多個文件。
也就是說, spl_autoload_register 是可以多次重復使用的,這一點正是解決了 __autoload 的短板,那么如果一個頁面有多個 spl_autoload_register ,執行順序是按照注冊的順序,一個一個往下找,如果找到了就停止。
3. spl_autoload_register自動加載和namespace命名空間
對于非常復雜的系統,其目錄結構也會非常復雜,規范的命名空間解決了復雜路徑下大量文件、函數、類重名的問題。而自動加載現在是PHP現代框架的基石,基本都是 spl_autoload_register 來實現自動加載。所以spl_autoload_register + namespace 就成為了一個主流。
根據PSR系列規范,namespace命名已經非常規范化,所以根據namespace就能找到詳細的路徑,從而找到類文件。
我們用最簡單的例子來說明復雜系統如何自動加載類文件。
首先,我們準備系統目錄結構:
- ----/Lib // 類目錄
- --Db.php
- --Say.php
- ----autoload.php // 自動加載函數
- ----index.php // 首頁
以上是一個基本的系統目錄,我們要實現的是,使用命名空間和自動加載,直接在首頁index.php調用Lib目錄下的多個類。
我們準備兩個列文件:
Db.php
- namespace Lib;
- class Db
- {
- public function __construct()
- {
- //echo 'Hello Db';
- }
- public static function test()
- {
- echo 'Test';
- }//Vevb.com
- }
Say.php
- namespace Lib;
- class Say
- {
- public function __construct()
- {
- //echo 'Hello';
- }
- public function hello()
- {
- echo 'say hello';
- }
- }
以上兩個普通的類文件,添加了命名空間: namespace Lib; 表示該類文件屬于Lib目錄名稱下的,當然你可以隨便取個不一樣的名字來表示你的項目名稱。
現在我們來看autoload.php:
- spl_autoload_register(function ($class) {
- $prefix = 'Lib//';
- $base_dir = __DIR__ . '/Lib/';
- // does the class use the namespace prefix?
- $len = strlen($prefix);
- if (strncmp($prefix, $class, $len) !== 0) {
- // no, move to the next registered autoloader
- return;
- }
- $relative_class = substr($class, $len);
- // 兼容Linux文件找。Windows 下(/ 和 /)是通用的
- $file = $base_dir . str_replace('//', '/', $relative_class) . '.php';
- if (file_exists($file)) {
- require $file;
- }
- });
以上代碼使用函數 spl_autoload_register() 首先判斷是否使用了命名空間,然后驗證要調用的類文件是否存在,如果存在就 require 類文件。
好了,現在我們在首頁index.php這樣調用:
- use Lib/Db;
- use Lib/Say;
- require './autoload.php';
- $db = new Db();
- $db::test();
- $say = new Say;
- $say->hello();
我們只需使用一個require將autoload.php加載進來,使用 use 關鍵字將類文件路徑變成絕對路徑了,當然你也可以在調用類的時候把路徑都寫上,如: new LibDb(); ,但是涉及到多個類互相調用的時候就會很棘手,所以我們還是在文件開頭就使用 use 把路徑處理好。
接下來就直接調用Lib/目錄下的各種類文件了,你可以在Lib/目錄下放置多個類文件嘗試下。
運行index.php看看是不是如您所愿。
結束語:
該文簡單介紹了自動加載以及命名空間的使用,實際開發中,我們很少去關注autoload自動加載的問題,因為大多數現代PHP框架都已經處理好了文件自動加載的問題。開發者只需關注業務代碼,使用規范的命名空間就可以了。當然,如果你想自己開發個項目不依賴大型框架亦或者自己開發php框架,那你就得熟悉下autoload自動加載這個好東西了,畢竟它可以讓我們“偷懶”,省事多了。
現代php里,我們經常使用 Composer 方式安裝的組件,都可以通過autoload實現自動加載,所以還是一個“懶”字給我們帶來了極好的開發效率。
新聞熱點
疑難解答