php的命名空間和自動加載實現

類的自動加載

引子

當咱們在php代碼中加載類時,咱們必需要include或者require 某個類文件。
但遇到相似的狀況,例如:php

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

假如咱們須要判斷一我的的性別,若是是男的就實例化class1這個類,若是是女的就實例化class2這個類。那麼問題來了:這段代碼,每次我只須要執行一個實例化對象,然而我必須加載這兩個類文件。html

php對於這種問題提出瞭解決方案laravel

spl_auto_register()

這個概念在 在php5.1中提出數組

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

函數包含3個參數

①autoload_function  這是一個函數【方法】名稱,能夠是字符串或者數組(調用類方法使用)。這個函數(方法)的功能就是,來把須要new 的類文件包含include(requeire)進來,這樣new的時候就不會找不到文件了。其實就是封裝整個項目的include和require功能。

② $throw 該參數指定當autoload_function沒法註冊時,spl_autoload_register()是否應引起異常。 

③ 若是爲true,那麼spl_autoload_register()將在自動加載到文件前面,而不時在它後面。

用法

那麼有了這個函數以後向這樣寫了php7

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

程序執行過程以下:mvc

// 正常的流程
new 一個對象-->找不到對象--> 報錯

// 引入spl_autoload_register 後
new 一個對象-->找不到對象--> spl_autoload_register對說交給我試試--> 加載成功

加載以後咱們執行了load這個函數,經過class的拼接,咱們完成了加載函數的過程app

__autoload()

類的自動加載在前面咱們講 spl_autoload_register 的時候已經和你們講過了。今天咱們講另外一種
__autoload() 在php7中已經不建議使用了框架

php的__autoload函數是一個魔術函數,在這個函數出現以前,若是一個php文件裏引用了100個對象,那麼這個文件就須要使用include或require引進100個類文件,這將致使該php文件無比龐大。因而就有了這個 __autoload函數。ide

__autoload函數在何時調用呢?當php文件中使用了new關鍵字實例化一個對象時,若是該類沒有在本php文件中被定義,將會觸發__autoload函數,此時,就能夠引進定義該類的php文件,然後,就能實例化成功了。函數

(注意:若是須要實例化的對象,在本文件中已經找到該類的定義的話,就不會觸發 __autoload 函數)

他和 spl_autoload_registe r的區別就在於當文件中同時出現__autoload和spl_autoload_register時,以spl_autoload_register爲準

命名空間

咱們先前講過類的自動加載,而後我就在思索。

咱們用框架寫代碼的時候,每在另外一個文件中調用其餘類時
咱們並無寫spl_autoload_register這個方法啊?那咱們時怎麼實現的呢?

原理

原來啊,咱們php在5.3時引入了命名空間的概念(這也是爲何大多數的框架不支持5.3以前的版本緣由之一),命名空間你們多少仍是瞭解的吧:不知道的去牆角面壁思過

命名空間簡而言之就是一種標識,它的主要目的是解決命名衝突的問題。就像在平常生活中,有不少姓名相同的人,如何區分這些人呢?那就須要加上一些額外的標識。把工做單位當成標識彷佛不錯,這樣就不用擔憂 「撞名」 的尷尬了。

命名空間分類

  • 徹底限定命名空間
  • 限定命名空間
new 成都\徐大帥(); // 限定類名
new \成都\徐大帥(); // 徹底限定類名

在當前命名空間沒有聲明的狀況下,限定類名和徹底限定類名是等價的。由於若是不指定空間,則默認爲全局()。

namespace 美國;

new 成都\徐大帥(); // 美國\成都\徐大帥(實際結果)
new \成都\徐大帥(); // 成都\徐大帥(實際結果)

這個例子展現了在命名空間下,使用限定類名和徹底限定類名的區別。(徹底限定類名 = 當前命名空間 + 限定類名)

/* 導入命名空間 */
use 成都\徐大帥;
new 徐大帥(); // 成都\徐大帥(實際結果)

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

/* 任何狀況 */
new \成都\徐大帥();// 成都\徐大帥(實際結果)

使用命名空間只是讓類名有了前綴,不容易發生衝突,系統仍然不會進行自動導入。

若是不引入文件,系統會在拋出 "Class Not Found" 錯誤以前觸發 __autoload() 或者spl_autoload_register函數,並將限定類名傳入做爲參數。

上面的例子都是基於你已經將相關文件手動引入的狀況下實現的,不然系統會拋出 " Class '成都徐大帥' not found"。由於她不知道這個文件在哪裏。因此在引入命名空間之後又引入了自動加載

接下來,咱們就在用命名空間加載咱們的 類

一個使用命名空間自動加載類的小實驗

首先,咱們在一個新文件中定義

//School.php
namespace top;

class School
{
    function __construct()
    {
        echo '這是'.__CLASS__.'類的實現';
    }
}

這固然不是重要的,重要的是咱們調用他的函數。咱們在同一個目錄創建一個index.php文件(不一樣文件也行,只要你寫好映射關係)

//index.php

spl_autoload_register(function ($class){
   //從咱們的 class名稱中找,有沒有對應的路徑
   $map = [
       'top\\School'=>'./School.php'
   ];

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

結果

開始
這是top\School類的實現

咱們使用了 類名和類地址的映射關係,實現了咱們的自動加載。然而這也意味着咱們每次添加文件,就必須去更新咱們的映射文件。在一個大型系統中這樣數組維持的映射關係無疑很麻煩。那麼有沒有好一點的作法呢?

PSR4 自動加載規範

不知道的童鞋,能夠看這裏
PSR4 中文文檔
PSR4 的具體解釋
下面摘自上面連接,我以爲上面兩篇文章已經講得很透徹了

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

PSR-4 規範中必需要有一個頂級命名空間,它的意義在於表示某一個特殊的目錄(文件基目錄)。子命名空間表明的是類文件相對於文件基目錄的這一段路徑(相對路徑),類名則與文件名保持一致(注意大小寫的區別)。

舉個例子:在全限定類名 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 能夠看出限定類名轉換爲路徑的過程。那麼如今就讓咱們用規範的面向對象方式去實現自動加載器吧。

首先咱們建立一個文件 Index.php,它處於 appmvcviewhome 目錄中:

namespace app\mvc\view\home;

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

接着咱們在建立一個加載類(不須要命名空間),它處於 目錄中:

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); // 文件標準路徑
    }

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

最後,將 Loader 類中的 autoload 註冊到 spl_autoload_register 函數中:

include 'Loader.php'; // 引入加載器
spl_autoload_register('Loader::autoload'); // 註冊自動加載

new \app\mvc\view\home\Index(); // 實例化未引用的類

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

示例中的代碼其實就是 ThinkPHP 自動加載器源碼的精簡版,它是 ThinkPHP 5 能實現惰性加載的關鍵。

相關文章
相關標籤/搜索