先跟大家說下什麼是命名空間。
「什麼是命名空間?從廣義上來說,命名空間是一種封裝事物的方法。在許多地方都可以見到這種抽象概念。例如,在作業系統中目錄用來將相關檔案分組,對於目錄中的文件來說,它就扮演了命名空間的角色。 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 <?php namespace 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实现自动加载,所以还是一个“懒”字给我们带来了极好的开发效率。
以上是PHP自動載入autoload和命名空間的應用程式小結的詳細內容。更多資訊請關注PHP中文網其他相關文章!