• <button id="miwm2"><object id="miwm2"></object></button>
  • <span id="miwm2"><pre id="miwm2"></pre></span>
      當前位置:源碼庫-建站教程-網站開發-PHP教程-php的命名空間與自動加載實現方法

      php的命名空間與自動加載實現方法

      時間:2019-08-25來源/作者:什么都只會一點點 編輯:源碼庫 文章熱度:

      類的自動加載

      引子

      當我們在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這個類。那么問題來了:這段代碼,每次我只需要執行一個實例化對象,然而我必須加載這兩個類文件。

      php對于這種問題提出了解決方案

      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()將在自動加載到文件前面,而不時在它后面。

      用法

      那么有了這個函數之后向這樣寫了

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

      程序執行過程如下:

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

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

      加載之后我們執行了load這個函數,通過class的拼接,我們完成了加載函數的過程

      __autoload()

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

      php的__autoload函數是一個魔術函數,在這個函數出現之前,如果一個php文件里引用了100個對象,那么這個文件就需要使用include或require引進100個類文件,這將導致該php文件無比龐大。于是就有了這個 __autoload函數。

      __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 能實現惰性加載的關鍵。

      總結

      以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,謝謝大家對ASPKU源碼庫的支持。


      注:相關教程知識閱讀請移步到PHP教程頻道。
      相關PHP教程
      熱門標簽

      PHP教程Rss訂閱PHP教程搜索

      撸吧撸吧