PHP自動(dòng)加載autoload和命名空間的應(yīng)用小結(jié)
先給大家說(shuō)下什么是命名空間。
“什么是命名空間?從廣義上來(lái)說(shuō),命名空間是一種封裝事物的方法。在很多地方都可以見(jiàn)到這種抽象概念。例如,在操作系統(tǒng)中目錄用來(lái)將相關(guān)文件分組,對(duì)于目錄中的文件來(lái)說(shuō),它就扮演了命名空間的角色。具體舉個(gè)例子,文件 foo.txt 可以同時(shí)在目錄/home/greg 和 /home/other 中存在,但在同一個(gè)目錄中不能存在兩個(gè) foo.txt 文件。另外,在目錄 /home/greg 外訪問(wèn) foo.txt 文件時(shí),我們必須將目錄名以及目錄分隔符放在文件名之前得到 /home/greg/foo.txt。這個(gè)原理應(yīng)用到程序設(shè)計(jì)領(lǐng)域就是命名空間的概念。”
PHP的自動(dòng)加載就是我們加載實(shí)例化類(lèi)的時(shí)候,不需要手動(dòng)去寫(xiě)require來(lái)導(dǎo)入這個(gè)class.php文件,程序自動(dòng)幫我們加載導(dǎo)入進(jìn)來(lái)。配合命名空間規(guī)范,我們可以在復(fù)雜系統(tǒng)中很輕松的處理不同類(lèi)的加載和調(diào)用問(wèn)題。
1. 自動(dòng)加載的原理以及__autoload的使用
自動(dòng)加載的原理,就是在我們實(shí)例化一個(gè) class 的時(shí)候,PHP如果找不到這個(gè)類(lèi),就會(huì)去自動(dòng)調(diào)用本文件中的 __autoload($class_name) 方法,我們new的這個(gè)class_name 就成為這個(gè)方法的參數(shù)。所以我們就可以在這個(gè)方法中根據(jù)我們需要new class_name的各種判斷和劃分就去require對(duì)應(yīng)的路徑類(lèi)文件,從而實(shí)現(xiàn)自動(dòng)加載。
我們先來(lái)看下 __autoload() 的自動(dòng)調(diào)用,舉個(gè)栗子:
index.php
<?php $db = new Db();
如果我們不手動(dòng)導(dǎo)入Db類(lèi),程序可能會(huì)報(bào)錯(cuò),說(shuō)找不到這個(gè)類(lèi):
Fatal error: Uncaught Error: Class 'DB' not found in D:\web\helloweba\demo\2017\autoload\index.php:2 Stack trace: #0 {main} thrown in D:\web\helloweba\demo\2017\autoload\index.php on line 2
那么,我們現(xiàn)在加入 __autoload() 這個(gè)方法再看看:
$db = new DB(); function __autoload($className) { echo $className; exit(); }
根據(jù)上面自動(dòng)加載機(jī)制的描述,會(huì)輸出:Db, 也就是我們需要new 的類(lèi)的類(lèi)名。所以,這個(gè)時(shí)候我們就可以在 __autoload() 方法里,根據(jù)需要去加載類(lèi)庫(kù)文件了。
2. spl_autoload_register自動(dòng)加載
如果是小項(xiàng)目,用 __autoload() 就能實(shí)現(xiàn)基本的自動(dòng)加載了。但是如果一個(gè)項(xiàng)目很大,或者需要不同的自動(dòng)加載來(lái)加載不同路徑的文件,這個(gè)時(shí)候__autoload就杯具了,因?yàn)橐粋€(gè)項(xiàng)目中只允許有一個(gè) __autoload() 函數(shù),因?yàn)?PHP 不允許函數(shù)重名了,也就是說(shuō)你不能聲明2個(gè) __autoload() 函數(shù)文件,否則會(huì)報(bào)致命錯(cuò)誤。那怎么辦呢?放心,你想到的,PHP大神早已經(jīng)想到。 所以 spl_autoload_register() 這樣又一個(gè)牛逼函數(shù)誕生了,并且取而代之它。它執(zhí)行效率更高,更靈活。
先看下它如何使用,在index.php中加入以下代碼。
<?php spl_autoload_register(function($className){ if (is_file('./Lib/' . $className . '.php')) { require './Lib/' . $className . '.php'; } }); $db = new Db(); $db::test();
在Lib\Db.php文件中加入以下代碼:
<?php class Db { public static function test() { echo 'Test'; } }
運(yùn)行index.php后,當(dāng)調(diào)用 new Db() 時(shí), spl_autoload_register 會(huì)自動(dòng)去lib/目錄下查找對(duì)應(yīng)的Db.php文件,成功后并且能夠執(zhí)行 $db::test(); 。同樣如果在Lib\目錄下有多個(gè)php類(lèi)文件,都可以在index.php中直接調(diào)用,而不需要使用 require 多個(gè)文件。
也就是說(shuō), spl_autoload_register 是可以多次重復(fù)使用的,這一點(diǎn)正是解決了 __autoload 的短板,那么如果一個(gè)頁(yè)面有多個(gè) spl_autoload_register ,執(zhí)行順序是按照注冊(cè)的順序,一個(gè)一個(gè)往下找,如果找到了就停止。
3. spl_autoload_register自動(dòng)加載和namespace命名空間
對(duì)于非常復(fù)雜的系統(tǒng),其目錄結(jié)構(gòu)也會(huì)非常復(fù)雜,規(guī)范的命名空間解決了復(fù)雜路徑下大量文件、函數(shù)、類(lèi)重名的問(wèn)題。而自動(dòng)加載現(xiàn)在是PHP現(xiàn)代框架的基石,基本都是 spl_autoload_register 來(lái)實(shí)現(xiàn)自動(dòng)加載。所以spl_autoload_register + namespace 就成為了一個(gè)主流。
根據(jù)PSR系列規(guī)范,namespace命名已經(jīng)非常規(guī)范化,所以根據(jù)namespace就能找到詳細(xì)的路徑,從而找到類(lèi)文件。
我們用最簡(jiǎn)單的例子來(lái)說(shuō)明復(fù)雜系統(tǒng)如何自動(dòng)加載類(lèi)文件。
首先,我們準(zhǔn)備系統(tǒng)目錄結(jié)構(gòu):
----/Lib // 類(lèi)目錄 --Db.php --Say.php ----autoload.php // 自動(dòng)加載函數(shù) ----index.php // 首頁(yè)
以上是一個(gè)基本的系統(tǒng)目錄,我們要實(shí)現(xiàn)的是,使用命名空間和自動(dòng)加載,直接在首頁(yè)index.php調(diào)用Lib目錄下的多個(gè)類(lèi)。
我們準(zhǔn)備兩個(gè)列文件:
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'; } }
以上兩個(gè)普通的類(lèi)文件,添加了命名空間: namespace Lib; 表示該類(lèi)文件屬于Lib\目錄名稱(chēng)下的,當(dāng)然你可以隨便取個(gè)不一樣的名字來(lái)表示你的項(xiàng)目名稱(chēng)。
現(xiàn)在我們來(lái)看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; } });
以上代碼使用函數(shù) spl_autoload_register() 首先判斷是否使用了命名空間,然后驗(yàn)證要調(diào)用的類(lèi)文件是否存在,如果存在就 require 類(lèi)文件。
好了,現(xiàn)在我們?cè)谑醉?yè)index.php這樣調(diào)用:
<?php use Lib\Db; use Lib\Say; require './autoload.php'; $db = new Db(); $db::test(); $say = new Say; $say->hello();
我們只需使用一個(gè)require將autoload.php加載進(jìn)來(lái),使用 use 關(guān)鍵字將類(lèi)文件路徑變成絕對(duì)路徑了,當(dāng)然你也可以在調(diào)用類(lèi)的時(shí)候把路徑都寫(xiě)上,如: new Lib\Db(); ,但是涉及到多個(gè)類(lèi)互相調(diào)用的時(shí)候就會(huì)很棘手,所以我們還是在文件開(kāi)頭就使用 use 把路徑處理好。
接下來(lái)就直接調(diào)用Lib/目錄下的各種類(lèi)文件了,你可以在Lib/目錄下放置多個(gè)類(lèi)文件嘗試下。
運(yùn)行index.php看看是不是如您所愿。
結(jié)束語(yǔ)
該文簡(jiǎn)單介紹了自動(dòng)加載以及命名空間的使用,實(shí)際開(kāi)發(fā)中,我們很少去關(guān)注autoload自動(dòng)加載的問(wèn)題,因?yàn)榇蠖鄶?shù)現(xiàn)代PHP框架都已經(jīng)處理好了文件自動(dòng)加載的問(wèn)題。開(kāi)發(fā)者只需關(guān)注業(yè)務(wù)代碼,使用規(guī)范的命名空間就可以了。當(dāng)然,如果你想自己開(kāi)發(fā)個(gè)項(xiàng)目不依賴(lài)大型框架亦或者自己開(kāi)發(fā)php框架,那你就得熟悉下autoload自動(dòng)加載這個(gè)好東西了,畢竟它可以讓我們“偷懶”,省事多了。
現(xiàn)代php里,我們經(jīng)常使用 Composer 方式安裝的組件,都可以通過(guò)autoload實(shí)現(xiàn)自動(dòng)加載,所以還是一個(gè)“懶”字給我們帶來(lái)了極好的開(kāi)發(fā)效率。
總結(jié)
以上所述是小編給大家介紹的PHP自動(dòng)加載autoload和命名空間的應(yīng)用小結(jié),希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
相關(guān)文章
JavaScript表單驗(yàn)證的兩種實(shí)現(xiàn)方法
這篇文章主要為大家詳細(xì)介紹了JavaScript表單驗(yàn)證的兩種實(shí)現(xiàn)方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-02-02JS格式化字符串的兩種方法(反引號(hào)與String.prototype)
本文一共介紹了兩種實(shí)現(xiàn)方式,使用反引號(hào)或自定義方法實(shí)現(xiàn),需要的朋友可以參考下2023-06-06javaScript window.event.keyCode 集合與測(cè)試方法
javaScript window.event.keyCode 集合,對(duì)于事件的代碼獲取可以用腳本監(jiān)聽(tīng)來(lái)實(shí)現(xiàn)。2010-05-05解決npm安裝Electron緩慢網(wǎng)絡(luò)超時(shí)導(dǎo)致失敗的問(wèn)題
下面小編就為大家分享一篇解決npm安裝Electron緩慢網(wǎng)絡(luò)超時(shí)導(dǎo)致失敗的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-02-02