php的命名空間與自動加載實現(xiàn)方法
類的自動加載
引子
當(dāng)我們在php代碼中加載類時,我們必須要include或者require 某個類文件。
但遇到類似的情況,例如:
require "Class1.php"; require "Class2.php"; $boy = $_GET['sex'] = 0?true:false; if($boy) { $class1 = new Class1(); }else{ $class2 = new Class2(); }
假如我們需要判斷一個人的性別,如果是男的就實例化class1這個類,如果是女的就實例化class2這個類。那么問題來了:這段代碼,每次我只需要執(zhí)行一個實例化對象,然而我必須加載這兩個類文件。
php對于這種問題提出了解決方案
spl_auto_register()
這個概念在 在php5.1中提出
spl_auto_register($autoload_function = null, $throw = true, $prepend = false)
函數(shù)包含3個參數(shù)
①autoload_function 這是一個函數(shù)【方法】名稱,可以是字符串或者數(shù)組(調(diào)用類方法使用)。這個函數(shù)(方法)的功能就是,來把需要new 的類文件包含include(requeire)進(jìn)來,這樣new的時候就不會找不到文件了。其實就是封裝整個項目的include和require功能。
② $throw 該參數(shù)指定當(dāng)autoload_function無法注冊時,spl_autoload_register()是否應(yīng)引發(fā)異常。
③ 如果為true,那么spl_autoload_register()將在自動加載到文件前面,而不時在它后面。
用法
那么有了這個函數(shù)之后向這樣寫了
function load($class) { require "./{$class}.php"; } spl_autoload_register('load'); if($boy) { $class1 = new Class1(); }else{ $class2 = new Class2(); }
程序執(zhí)行過程如下:
// 正常的流程
new 一個對象-->找不到對象--> 報錯// 引入spl_autoload_register 后
new 一個對象-->找不到對象--> spl_autoload_register對說交給我試試--> 加載成功
加載之后我們執(zhí)行了load這個函數(shù),通過class的拼接,我們完成了加載函數(shù)的過程
__autoload()
類的自動加載在前面我們講 spl_autoload_register 的時候已經(jīng)和大家講過了。今天我們講另一種
__autoload() 在php7中已經(jīng)不建議使用了
php的__autoload函數(shù)是一個魔術(shù)函數(shù),在這個函數(shù)出現(xiàn)之前,如果一個php文件里引用了100個對象,那么這個文件就需要使用include或require引進(jìn)100個類文件,這將導(dǎo)致該php文件無比龐大。于是就有了這個 __autoload函數(shù)。
__autoload函數(shù)在什么時候調(diào)用呢?當(dāng)php文件中使用了new關(guān)鍵字實例化一個對象時,如果該類沒有在本php文件中被定義,將會觸發(fā)__autoload函數(shù),此時,就可以引進(jìn)定義該類的php文件,而后,就能實例化成功了。
(注意:如果需要實例化的對象,在本文件中已經(jīng)找到該類的定義的話,就不會觸發(fā) __autoload 函數(shù))
他和 spl_autoload_registe r的區(qū)別就在于當(dāng)文件中同時出現(xiàn)__autoload和spl_autoload_register時,以spl_autoload_register為準(zhǔn)
命名空間
我們先前講過類的自動加載,然后我就在思索。
我們用框架寫代碼的時候,每在另一個文件中調(diào)用其他類時
我們并沒有寫spl_autoload_register這個方法???那我們時怎么實現(xiàn)的呢?
原理
原來啊,我們php在5.3時引入了命名空間的概念(這也是為什么大多數(shù)的框架不支持5.3之前的版本原因之一),命名空間大家多少還是了解的吧:不知道的去墻角面壁思過
命名空間簡而言之就是一種標(biāo)識,它的主要目的是解決命名沖突的問題。就像在日常生活中,有很多姓名相同的人,如何區(qū)分這些人呢?那就需要加上一些額外的標(biāo)識。把工作單位當(dāng)成標(biāo)識似乎不錯,這樣就不用擔(dān)心 “撞名” 的尷尬了。
命名空間分類
- 完全限定命名空間
- 限定命名空間
new 成都\徐大帥(); // 限定類名 new \成都\徐大帥(); // 完全限定類名
在當(dāng)前命名空間沒有聲明的情況下,限定類名和完全限定類名是等價的。因為如果不指定空間,則默認(rèn)為全局()。
namespace 美國; new 成都\徐大帥(); // 美國\成都\徐大帥(實際結(jié)果) new \成都\徐大帥(); // 成都\徐大帥(實際結(jié)果)
這個例子展示了在命名空間下,使用限定類名和完全限定類名的區(qū)別。(完全限定類名 = 當(dāng)前命名空間 + 限定類名)
/* 導(dǎo)入命名空間 */ use 成都\徐大帥; new 徐大帥(); // 成都\徐大帥(實際結(jié)果) /* 設(shè)置別名 */ use 成都\徐大帥 AS CEO; new CEO(); // 成都\徐大帥(實際結(jié)果) /* 任何情況 */ new \成都\徐大帥();// 成都\徐大帥(實際結(jié)果)
使用命名空間只是讓類名有了前綴,不容易發(fā)生沖突,系統(tǒng)仍然不會進(jìn)行自動導(dǎo)入。
如果不引入文件,系統(tǒng)會在拋出 "Class Not Found" 錯誤之前觸發(fā) __autoload() 或者spl_autoload_register函數(shù),并將限定類名傳入作為參數(shù)。
上面的例子都是基于你已經(jīng)將相關(guān)文件手動引入的情況下實現(xiàn)的,否則系統(tǒng)會拋出 " Class '成都徐大帥' not found"。因為她不知道這個文件在哪里。所以在引入命名空間以后又引入了自動加載
接下來,我們就在用命名空間加載我們的 類
一個使用命名空間自動加載類的小實驗
首先,我們在一個新文件中定義
//School.php namespace top; class School { function __construct() { echo '這是'.__CLASS__.'類的實現(xiàn)'; } }
這當(dāng)然不是重要的,重要的是我們調(diào)用他的函數(shù)。我們在同一個目錄建立一個index.php文件(不同文件也行,只要你寫好映射關(guān)系)
//index.php spl_autoload_register(function ($class){ //從我們的 class名稱中找,有沒有對應(yīng)的路徑 $map = [ 'top\\School'=>'./School.php' ]; $file = $map[$class]; //查看對應(yīng)的文件是否存在 if (file_exists($file)) include $file; }); echo "開始<br/>"; new top\School();
結(jié)果
開始
這是top\School類的實現(xiàn)
我們使用了 類名和類地址的映射關(guān)系,實現(xiàn)了我們的自動加載。然而這也意味著我們每次添加文件,就必須去更新我們的映射文件。在一個大型系統(tǒng)中這樣數(shù)組維持的映射關(guān)系無疑很麻煩。那么有沒有好一點的做法呢?
PSR4 自動加載規(guī)范
不知道的童鞋,可以看這里
下面摘自上面鏈接,我覺得上面兩篇文章已經(jīng)講得很透徹了
\<NamespaceName>(\<SubNamespaceNames>)*\<ClassName>
PSR-4 規(guī)范中必須要有一個頂級命名空間,它的意義在于表示某一個特殊的目錄(文件基目錄)。子命名空間代表的是類文件相對于文件基目錄的這一段路徑(相對路徑),類名則與文件名保持一致(注意大小寫的區(qū)別)。
舉個例子:在全限定類名 appviewnewsIndex 中,如果 app 代表 C:Baidu,那么這個類的路徑則是 C:BaiduviewnewsIndex.php
我們就以解析 appviewnewsIndex 為例,編寫一個簡單的 Demo:
$class = 'app\view\news\Index'; /* 頂級命名空間路徑映射 */ $vendor_map = array( 'app' => 'C:\Baidu', ); /* 解析類名為文件路徑 */ $vendor = substr($class, 0, strpos($class, '\\')); // 取出頂級命名空間[app] $vendor_dir = $vendor_map[$vendor]; // 文件基目錄[C:\Baidu] $rel_path = dirname(substr($class, strlen($vendor))); // 相對路徑[/view/news] $file_name = basename($class) . '.php'; // 文件名[Index.php] /* 輸出文件所在路徑 */ echo $vendor_dir . $rel_path . DIRECTORY_SEPARATOR . $file_name;
通過這個 Demo 可以看出限定類名轉(zhuǎn)換為路徑的過程。那么現(xiàn)在就讓我們用規(guī)范的面向?qū)ο蠓绞饺崿F(xiàn)自動加載器吧。
首先我們創(chuàng)建一個文件 Index.php,它處于 appmvcviewhome 目錄中:
namespace app\mvc\view\home; class Index { function __construct() { echo '<h1> Welcome To Home </h1>'; } }
接著我們在創(chuàng)建一個加載類(不需要命名空間),它處于 目錄中:
class Loader { /* 路徑映射 */ public static $vendorMap = array( 'app' => __DIR__ . DIRECTORY_SEPARATOR . 'app', ); /** * 自動加載器 */ public static function autoload($class) { $file = self::findFile($class); if (file_exists($file)) { self::includeFile($file); } } /** * 解析文件路徑 */ private static function findFile($class) { $vendor = substr($class, 0, strpos($class, '\\')); // 頂級命名空間 $vendorDir = self::$vendorMap[$vendor]; // 文件基目錄 $filePath = substr($class, strlen($vendor)) . '.php'; // 文件相對路徑 return strtr($vendorDir . $filePath, '\\', DIRECTORY_SEPARATOR); // 文件標(biāo)準(zhǔn)路徑 } /** * 引入文件 */ private static function includeFile($file) { if (is_file($file)) { include $file; } } }
最后,將 Loader 類中的 autoload 注冊到 spl_autoload_register 函數(shù)中:
include 'Loader.php'; // 引入加載器 spl_autoload_register('Loader::autoload'); // 注冊自動加載 new \app\mvc\view\home\Index(); // 實例化未引用的類 /** * 輸出: <h1> Welcome To Home </h1> */
示例中的代碼其實就是 ThinkPHP 自動加載器源碼的精簡版,它是 ThinkPHP 5 能實現(xiàn)惰性加載的關(guān)鍵。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,謝謝大家對腳本之家的支持。
相關(guān)文章
PHP使用imagick擴(kuò)展實現(xiàn)合并圖像的方法
這篇文章主要介紹了PHP使用imagick擴(kuò)展實現(xiàn)合并圖像的方法,結(jié)合實例形式分析了php基于imagick擴(kuò)展處理圖片的具體步驟與相關(guān)操作技巧,需要的朋友可以參考下2017-04-04php獲取一定范圍內(nèi)取N個不重復(fù)的隨機(jī)數(shù)
這篇文章主要介紹了php獲取一定范圍內(nèi)取N個不重復(fù)的隨機(jī)數(shù)的方法,通過range函數(shù)創(chuàng)建指定范圍內(nèi)數(shù)組及shuffle進(jìn)行數(shù)組隨機(jī)排序,并使用array_slice抽取數(shù)組實現(xiàn)該功能,非常簡單實用,需要的朋友可以參考下2016-05-05Mac M1安裝mnmp(Mac+Nginx+MySQL+PHP)開發(fā)環(huán)境
這篇文章主要介紹了Mac M1安裝mnmp(Mac+Nginx+MySQL+PHP)開發(fā)環(huán)境,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-03-03PHP單例模式Singleton Pattern的原理與實現(xiàn)介紹
單例就是單實例的意思,即在系統(tǒng)全局,一個類只創(chuàng)建一個對象,并且在系統(tǒng)全局都可以訪問這個對象而不用重新創(chuàng)建。本文將通過示例為大家詳細(xì)講解Java單例模式的使用,需要的可以參考一下2023-03-03PHP使用DirectoryIterator顯示下拉文件列表的方法
這篇文章主要介紹了PHP使用DirectoryIterator顯示下拉文件列表的方法,涉及php使用DirectoryIterator操作文件的技巧,需要的朋友可以參考下2015-03-03PHP 服務(wù)器配置(使用Apache及IIS兩種方法)
簡單介紹使用Apache及IIS解析PHP2009-06-06