先給大家說下什麼是命名空間。
“什麼是命名空間?從廣義上來說,命名空間是一種封裝事物的方法。在很多地方都可以見到這種抽象概念。例如,在作業系統中目錄用來將相關檔案分組,對於目錄中的檔案來說,它就扮演了命名空間的角色。具體舉個例子,檔案 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
<?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中加入以下代碼。
<?php spl_autoload_register(function($className){ if (is_file('./Lib/' . $className . '.php')) { require './Lib/' . $className . '.php'; }});$db = new Db();$db::test();
在LibDb.php檔案中加入以下代碼:
<?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
<?php namespace Lib;class Db{ public function __construct() { //echo 'Hello Db'; } public static function test() { echo 'Test'; }}Say.php<?phpnamespace Lib;class Say { public function __construct() { //echo 'Hello'; } public function hello() { echo 'say hello'; }}
以上兩個普通的類檔案,添加了命名空間: namespace Lib; 表示該類檔案屬於Lib目錄名稱下的,當然你可以隨便取個不一樣的名字來表示你的項目名稱。
現在我們來看autoload.php:
<?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這樣調用:
<?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實現自動載入,所以還是一個“懶”字給我們帶來了極好的開發效率。