溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶(hù)服務(wù)條款》

怎么在php項(xiàng)目中實(shí)現(xiàn)一個(gè)命名空間與自動(dòng)加載功能

發(fā)布時(shí)間:2021-02-08 15:28:58 來(lái)源:億速云 閱讀:175 作者:Leah 欄目:開(kāi)發(fā)技術(shù)

本篇文章給大家分享的是有關(guān)怎么在php項(xiàng)目中實(shí)現(xiàn)一個(gè)命名空間與自動(dòng)加載功能,小編覺(jué)得挺實(shí)用的,因此分享給大家學(xué)習(xí),希望大家閱讀完這篇文章后可以有所收獲,話(huà)不多說(shuō),跟著小編一起來(lái)看看吧。

require "Class1.php";
require "Class2.php";
$boy = $_GET['sex'] = 0?true:false;
if($boy)
{
 $class1 = new Class1();
}else{
 $class2 = new Class2();
}

假如我們需要判斷一個(gè)人的性別,如果是男的就實(shí)例化class1這個(gè)類(lèi),如果是女的就實(shí)例化class2這個(gè)類(lèi)。那么問(wèn)題來(lái)了:這段代碼,每次我只需要執(zhí)行一個(gè)實(shí)例化對(duì)象,然而我必須加載這兩個(gè)類(lèi)文件。

php對(duì)于這種問(wèn)題提出了解決方案

spl_auto_register()

這個(gè)概念在 在php5.1中提出

spl_auto_register($autoload_function = null, $throw = true, $prepend = false)

函數(shù)包含3個(gè)參數(shù)

①autoload_function  這是一個(gè)函數(shù)【方法】名稱(chēng),可以是字符串或者數(shù)組(調(diào)用類(lèi)方法使用)。這個(gè)函數(shù)(方法)的功能就是,來(lái)把需要new 的類(lèi)文件包含include(requeire)進(jìn)來(lái),這樣new的時(shí)候就不會(huì)找不到文件了。其實(shí)就是封裝整個(gè)項(xiàng)目的include和require功能。

② $throw 該參數(shù)指定當(dāng)autoload_function無(wú)法注冊(cè)時(shí),spl_autoload_register()是否應(yīng)引發(fā)異常。

③ 如果為true,那么spl_autoload_register()將在自動(dòng)加載到文件前面,而不時(shí)在它后面。

用法

那么有了這個(gè)函數(shù)之后向這樣寫(xiě)了

function load($class)
{
 require "./{$class}.php";
}
spl_autoload_register('load');
if($boy)
{
 $class1 = new Class1();
}else{
 $class2 = new Class2();
}

程序執(zhí)行過(guò)程如下:

// 正常的流程
new 一個(gè)對(duì)象-->找不到對(duì)象--> 報(bào)錯(cuò)

// 引入spl_autoload_register 后
new 一個(gè)對(duì)象-->找不到對(duì)象--> spl_autoload_register對(duì)說(shuō)交給我試試--> 加載成功

加載之后我們執(zhí)行了load這個(gè)函數(shù),通過(guò)class的拼接,我們完成了加載函數(shù)的過(guò)程

__autoload()

類(lèi)的自動(dòng)加載在前面我們講 spl_autoload_register 的時(shí)候已經(jīng)和大家講過(guò)了。今天我們講另一種
__autoload() 在php7中已經(jīng)不建議使用了

php的__autoload函數(shù)是一個(gè)魔術(shù)函數(shù),在這個(gè)函數(shù)出現(xiàn)之前,如果一個(gè)php文件里引用了100個(gè)對(duì)象,那么這個(gè)文件就需要使用include或require引進(jìn)100個(gè)類(lèi)文件,這將導(dǎo)致該php文件無(wú)比龐大。于是就有了這個(gè) __autoload函數(shù)。

__autoload函數(shù)在什么時(shí)候調(diào)用呢?當(dāng)php文件中使用了new關(guān)鍵字實(shí)例化一個(gè)對(duì)象時(shí),如果該類(lèi)沒(méi)有在本php文件中被定義,將會(huì)觸發(fā)__autoload函數(shù),此時(shí),就可以引進(jìn)定義該類(lèi)的php文件,而后,就能實(shí)例化成功了。

(注意:如果需要實(shí)例化的對(duì)象,在本文件中已經(jīng)找到該類(lèi)的定義的話(huà),就不會(huì)觸發(fā) __autoload 函數(shù))

他和 spl_autoload_registe r的區(qū)別就在于當(dāng)文件中同時(shí)出現(xiàn)__autoload和spl_autoload_register時(shí),以spl_autoload_register為準(zhǔn)

命名空間

我們先前講過(guò)類(lèi)的自動(dòng)加載,然后我就在思索。

我們用框架寫(xiě)代碼的時(shí)候,每在另一個(gè)文件中調(diào)用其他類(lèi)時(shí)

我們并沒(méi)有寫(xiě)spl_autoload_register這個(gè)方法啊?那我們時(shí)怎么實(shí)現(xiàn)的呢?

原理

原來(lái)啊,我們php在5.3時(shí)引入了命名空間的概念(這也是為什么大多數(shù)的框架不支持5.3之前的版本原因之一),命名空間大家多少還是了解的吧:不知道的去墻角面壁思過(guò)

命名空間簡(jiǎn)而言之就是一種標(biāo)識(shí),它的主要目的是解決命名沖突的問(wèn)題。就像在日常生活中,有很多姓名相同的人,如何區(qū)分這些人呢?那就需要加上一些額外的標(biāo)識(shí)。把工作單位當(dāng)成標(biāo)識(shí)似乎不錯(cuò),這樣就不用擔(dān)心 “撞名” 的尷尬了。

命名空間分類(lèi)

  • 完全限定命名空間

  • 限定命名空間

new 成都\徐大帥(); // 限定類(lèi)名
new \成都\徐大帥(); // 完全限定類(lèi)名

在當(dāng)前命名空間沒(méi)有聲明的情況下,限定類(lèi)名和完全限定類(lèi)名是等價(jià)的。因?yàn)槿绻恢付臻g,則默認(rèn)為全局()。

namespace 美國(guó);

new 成都\徐大帥(); // 美國(guó)\成都\徐大帥(實(shí)際結(jié)果)
new \成都\徐大帥(); // 成都\徐大帥(實(shí)際結(jié)果)

這個(gè)例子展示了在命名空間下,使用限定類(lèi)名和完全限定類(lèi)名的區(qū)別。(完全限定類(lèi)名 = 當(dāng)前命名空間 + 限定類(lèi)名)

/* 導(dǎo)入命名空間 */
use 成都\徐大帥;
new 徐大帥(); // 成都\徐大帥(實(shí)際結(jié)果)

/* 設(shè)置別名 */
use 成都\徐大帥 AS CEO;
new CEO(); // 成都\徐大帥(實(shí)際結(jié)果)

/* 任何情況 */
new \成都\徐大帥();// 成都\徐大帥(實(shí)際結(jié)果)

使用命名空間只是讓類(lèi)名有了前綴,不容易發(fā)生沖突,系統(tǒng)仍然不會(huì)進(jìn)行自動(dòng)導(dǎo)入。

如果不引入文件,系統(tǒng)會(huì)在拋出 "Class Not Found" 錯(cuò)誤之前觸發(fā) __autoload() 或者spl_autoload_register函數(shù),并將限定類(lèi)名傳入作為參數(shù)。

上面的例子都是基于你已經(jīng)將相關(guān)文件手動(dòng)引入的情況下實(shí)現(xiàn)的,否則系統(tǒng)會(huì)拋出 " Class '成都徐大帥' not found"。因?yàn)樗恢肋@個(gè)文件在哪里。所以在引入命名空間以后又引入了自動(dòng)加載

接下來(lái),我們就在用命名空間加載我們的 類(lèi)

一個(gè)使用命名空間自動(dòng)加載類(lèi)的小實(shí)驗(yàn)

首先,我們?cè)谝粋€(gè)新文件中定義

//School.php
namespace top;

class School
{
 function __construct()
 {
 echo '這是'.__CLASS__.'類(lèi)的實(shí)現(xiàn)';
 }
}

這當(dāng)然不是重要的,重要的是我們調(diào)用他的函數(shù)。我們?cè)谕粋€(gè)目錄建立一個(gè)index.php文件(不同文件也行,只要你寫(xiě)好映射關(guān)系)

//index.php

spl_autoload_register(function ($class){
 //從我們的 class名稱(chēng)中找,有沒(méi)有對(duì)應(yīng)的路徑
 $map = [
 'top\\School'=>'./School.php'
 ];

 $file = $map[$class];
 //查看對(duì)應(yīng)的文件是否存在
 if (file_exists($file))
 include $file;
});
echo "開(kāi)始<br/>";
new top\School();

結(jié)果

開(kāi)始
這是top\School類(lèi)的實(shí)現(xiàn)

我們使用了 類(lèi)名和類(lèi)地址的映射關(guān)系,實(shí)現(xiàn)了我們的自動(dòng)加載。然而這也意味著我們每次添加文件,就必須去更新我們的映射文件。在一個(gè)大型系統(tǒng)中這樣數(shù)組維持的映射關(guān)系無(wú)疑很麻煩。那么有沒(méi)有好一點(diǎn)的做法呢?

PSR4 自動(dòng)加載規(guī)范

不知道的童鞋,可以看這里

PSR4 中文文檔

PSR4 的具體解釋

下面摘自上面鏈接,我覺(jué)得上面兩篇文章已經(jīng)講得很透徹了

\<NamespaceName>(\<SubNamespaceNames>)*\<ClassName>

PSR-4 規(guī)范中必須要有一個(gè)頂級(jí)命名空間,它的意義在于表示某一個(gè)特殊的目錄(文件基目錄)。子命名空間代表的是類(lèi)文件相對(duì)于文件基目錄的這一段路徑(相對(duì)路徑),類(lèi)名則與文件名保持一致(注意大小寫(xiě)的區(qū)別)。

舉個(gè)例子:在全限定類(lèi)名 appviewnewsIndex 中,如果 app 代表 C:Baidu,那么這個(gè)類(lèi)的路徑則是 C:BaiduviewnewsIndex.php

我們就以解析 appviewnewsIndex 為例,編寫(xiě)一個(gè)簡(jiǎn)單的 Demo:

$class = 'app\view\news\Index';

/* 頂級(jí)命名空間路徑映射 */
$vendor_map = array(
 'app' => 'C:\Baidu',
);

/* 解析類(lèi)名為文件路徑 */
$vendor = substr($class, 0, strpos($class, '\\')); // 取出頂級(jí)命名空間[app]
$vendor_dir = $vendor_map[$vendor]; // 文件基目錄[C:\Baidu]
$rel_path = dirname(substr($class, strlen($vendor))); // 相對(duì)路徑[/view/news]
$file_name = basename($class) . '.php'; // 文件名[Index.php]

/* 輸出文件所在路徑 */
echo $vendor_dir . $rel_path . DIRECTORY_SEPARATOR . $file_name;

通過(guò)這個(gè) Demo 可以看出限定類(lèi)名轉(zhuǎn)換為路徑的過(guò)程。那么現(xiàn)在就讓我們用規(guī)范的面向?qū)ο蠓绞饺?shí)現(xiàn)自動(dòng)加載器吧。

首先我們創(chuàng)建一個(gè)文件 Index.php,它處于 appmvcviewhome 目錄中:

namespace app\mvc\view\home;

class Index
{
 function __construct()
 {
  echo '<h2> Welcome To Home </h2>';
 }
}

接著我們?cè)趧?chuàng)建一個(gè)加載類(lèi)(不需要命名空間),它處于 目錄中:

class Loader
{
 /* 路徑映射 */
 public static $vendorMap = array(
  'app' => __DIR__ . DIRECTORY_SEPARATOR . 'app',
 );

 /**
  * 自動(dòng)加載器
  */
 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, '\\')); // 頂級(jí)命名空間
  $vendorDir = self::$vendorMap[$vendor]; // 文件基目錄
  $filePath = substr($class, strlen($vendor)) . '.php'; // 文件相對(duì)路徑
  return strtr($vendorDir . $filePath, '\\', DIRECTORY_SEPARATOR); // 文件標(biāo)準(zhǔn)路徑
 }

 /**
  * 引入文件
  */
 private static function includeFile($file)
 {
  if (is_file($file)) {
   include $file;
  }
 }
}

最后,將 Loader 類(lèi)中的 autoload 注冊(cè)到 spl_autoload_register 函數(shù)中:

include 'Loader.php'; // 引入加載器
spl_autoload_register('Loader::autoload'); // 注冊(cè)自動(dòng)加載

new \app\mvc\view\home\Index(); // 實(shí)例化未引用的類(lèi)

/**
 * 輸出: <h2> Welcome To Home </h2>
 */

以上就是怎么在php項(xiàng)目中實(shí)現(xiàn)一個(gè)命名空間與自動(dòng)加載功能,小編相信有部分知識(shí)點(diǎn)可能是我們?nèi)粘9ぷ鲿?huì)見(jiàn)到或用到的。希望你能通過(guò)這篇文章學(xué)到更多知識(shí)。更多詳情敬請(qǐng)關(guān)注億速云行業(yè)資訊頻道。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀(guān)點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

php
AI