<ruby id="bdb3f"></ruby>

    <p id="bdb3f"><cite id="bdb3f"></cite></p>

      <p id="bdb3f"><cite id="bdb3f"><th id="bdb3f"></th></cite></p><p id="bdb3f"></p>
        <p id="bdb3f"><cite id="bdb3f"></cite></p>

          <pre id="bdb3f"></pre>
          <pre id="bdb3f"><del id="bdb3f"><thead id="bdb3f"></thead></del></pre>

          <ruby id="bdb3f"><mark id="bdb3f"></mark></ruby><ruby id="bdb3f"></ruby>
          <pre id="bdb3f"><pre id="bdb3f"><mark id="bdb3f"></mark></pre></pre><output id="bdb3f"></output><p id="bdb3f"></p><p id="bdb3f"></p>

          <pre id="bdb3f"><del id="bdb3f"><progress id="bdb3f"></progress></del></pre>

                <ruby id="bdb3f"></ruby>

                合規國際互聯網加速 OSASE為企業客戶提供高速穩定SD-WAN國際加速解決方案。 廣告
                ## **自動加載** * 什么是自動加載?簡單描述一下就是:自動加載就是我們再new一個class的時候,不需要手動require來導入這個class.php文件,程序自動幫我們加載導入進來。這是php5.1.2(好像是)版本新加入的一個功能。他解放了程序員的雙手,不需要手動寫那么多require,變得有那么點智能的感覺。 * 自動加載可以說是現代php框架的根基,任務厲害的框架或者架構都會用到它,它發明的理由是啥呢?一個字懶。因為項目越來越大,相關聯的類庫文件就越來越多。我們不可能像小項目那樣用require那樣全部手動一個一個的require。 ## **自動加載的原理以及__autoload的使用** * 自動加載的原理,就是我們在new一個class的時候,php系統如果找不到這個類,就會去自動調用本文件中的autoload($class_name)方法,我們new的這個class_name就成為這個方法的參數。所以我們就可以在這個方法中根據我們需要的new class_name的各種判斷和劃分去require對應的路徑類文件,從而實現自動加載。 我們先一步一步來,看下__autoload()的自動調用,看個例子: index.php ``` $db = new DB(); ``` 如果我們不手動導入DB類,程序可能會報錯,說找不到這個類: > Fatal error: Class 'DB' not found in D:\\wamp\\www\\testphp\\autoload\\index.php on line 3 那么,我們再加入__autoload()方法里了,根據需要去加載類庫文件了 index.php ``` $db =new DB(); function __autoload($className) { echo $className; exit(); } ``` 根據上面自動加載機制的描述,你分析下會輸出什么? 沒錯:肯定是輸出:DB, 也就是我們需要`new`的類的類名。所以,這個時候我們就可以在`__autoload()`方法里,根據需要去加載類庫文件了。 index.php ``` $db =new DB(); function __autoload($className) { require $className .'.php'; } ``` DB.php ``` class DB { publicfunction __construct() { echo 'Hello DB'; } } ``` 這樣子我們就很輕松的將我們需要`new`的class 全部導入了進來,這樣子,我們就可以輕松的`new`N個class,比如: ``` <?php function __autoload($className) { require $className .'.php'; } $db =new DB(); $info =newInfo(); $gender =newGender(); $name =newName(); //也是支持靜態方法直接調用的 Height::test(); ``` ## **spl_autoload_register的使用** 小的項目,用__autoload()就能實現基本的自動加載了。但是如果一個項目過大,或者需要不同的自動加載來加載不同路徑的文件,這個時候__autoload就悲劇了,原因是一個項目中僅能有一個這樣的 __autoload() 函數,因為 PHP 不允許函數重名,也就是說你不能聲明2個__autoload()函數文件,否則會報致命錯誤,我了個大擦,那怎么辦呢?放心,你想到的,PHP開發大神早已經想到。 所以spl_autoload_register()這樣又一個牛逼函數誕生了,并且取而代之它。它執行效率更高,更靈活 先看下它如何使用吧: 當我們去new一個找不到的class時,PHP就會去自動調用sql_autoload_resister注冊的函數,這個函數通過它的參數傳進去: ``` sql_autoload_resister($param) 這個參數可以有多種形式: sql_autoload_resister('load_function'); //函數名 sql_autoload_resister(array('load_object', 'load_function')); //類和靜態方法 sql_autoload_resister('load_object::load_function'); //類和方法的靜態調用 //php 5.3之后,也可以像這樣支持匿名函數了。 spl_autoload_register(function($className){ if (is_file('./lib/' . $className . '.php')) { require './lib/' . $className . '.php'; } }); ``` index.php ``` function load1($className) { echo 1; require $className .'.php'; } spl_autoload_register('load1');//將load1函數注冊到自動加載隊列中。 $db =new DB();//找不到DB類,就會自動去調用剛注冊的load1函數了 ``` 上面就是實現了自動加載的方式,我們同樣也可以用類加載的方式調用,但是必須是static方法: ``` class autoloading { //必須是靜態方法,不然報錯 public static function load($className) { require $className .'.php'; } } //2種方法都可以 spl_autoload_register(array('autoloading','load')); spl_autoload_register('autoloading::load'); $db =new DB();//會自動找到 ``` 需要注意的是,如果你同時使用spl_autoload_register和__autoload,__autoload會失效!!! 再說了,本來就是替換它的,就一心使用spl_autoload_register就好了。 ## **多個spl_autoload_register的使用** spl_autoload_register是可以多次重復使用的,這一點正是解決了__autoload的短板,那么如果一個頁面有多個,執行順序是按照注冊的順序,一個一個往下找,如果找到了就停止。 我們來看下這個例子,DB.php就在本目錄下,Info.php在/lib/目錄下。 ``` function load1($className) { echo 1; if(is_file($className .'.php')){ require $className .'.php'; } } function load2($className) { echo 2; if(is_file('./app/'. $className .'.php')){ require'./app/'. $className .'.php'; } } function __autoload($className) { echo 3; if(is_file('./lib/'. $className .'.php')){ require'./lib/'. $className .'.php'; } } //注冊了3個 spl_autoload_register('load1'); spl_autoload_register('load2'); spl_autoload_register('__autoload'); $db =new DB();//DB就在本目錄下 $info =newInfo();//Info 在/lib/Info.php ``` 我們注冊了3個自動加載函數。執行結果是啥呢? > 1Hello DB > 123Hello Info > 我們分析下: new DB的時候,就按照注冊順序,先去找load1()函數了,發現找到了,就停止了,所以輸出1 Hello Word new Info的時候,先是安裝注冊順序,先找load1(), 所以輸出了1,發現沒找到,就去load2()里面去找,所以輸出了2,還是沒這個文件,就去__autoload()函數里找,所以,先輸出了3,再輸出Hello Info 注意,前面說過,spl_autoload_register使用時,__autoload會無效,有時候,我們希望它繼續有效,就可以也將它注冊進來,就可以繼續使用。 我們可以打印spl_autoload_functions()函數,來顯示一共注冊了多少個自動加載: ``` var_dump(spl_autoload_functions()); //數組的形式輸出 array (size=3) 0 => string 'load1' (length=5) 1 => string 'load2' (length=5) 2 => string '__autoload' (length=10) ``` ## **spl_autoload_register自動加載+namespace命名空間 的使用** 前面已經說過,自動加載現在是PHP現代框架的基石,基本都是spl_autoload_register來實現自動加載。namespace也是使用比較多的。所以spl_autoload_register + namespace 就成為了一個主流。根據PSR-0的規范,namespace命名已經非常規范化,所以用namespace就能找到詳細的路徑,從而找到類文件。 我們舉例子來看下: AutoLoading\loading ``` <?php namespaceAutoLoading; class loading { public static function autoload($className) { //根據PSR-O的第4點 把 \ 轉換層(目錄風格符) DIRECTORY_SEPARATOR , //便于兼容Linux文件找。Windows 下(/ 和 \)是通用的 //由于namspace 很規格,所以直接很快就能找到 $fileName = str_replace('\\', DIRECTORY_SEPARATOR, DIR .'\\'. $className).'.php'; if(is_file($fileName)){ require $fileName; }else{ echo $fileName .' is not exist';die; } } } ``` 上面就是一個自動加載的核心思想方法。下面我們就來spl_autoload_register來注冊這個函數: index.php ``` <?php //定義當前的目錄絕對路徑 define('DIR', dirname(__FILE__)); //加載這個文件 require DIR .'/loading.php'; //采用`命名空間`的方式注冊。php 5.3 加入的 //也必須是得是static靜態方法調用,然后就像加載namespace的方式調用,注意:不能使用use spl_autoload_register("\\AutoLoading\\loading::autoload"); // 調用三個namespace類 //定位到Lib目錄下的Name.php Lib\Name::test(); //定位到App目錄下Android目錄下的Name.php App\Android\Name::test(); //定位到App目錄下Ios目錄下的Name.php App\Ios\Name::test(); ``` 由于我們是采用PSR-O方式來定義namespace的命名的,所以很好的定位到這個文件的在哪個目錄下了。很爽。對不對。 APP\Android\Name ``` <?php namespaceApp\Android; className { public function __construct() { echo __NAMESPACE__ ."<br>"; } public static function test() { echo __NAMESPACE__ .' static function test <br>'; } } ``` 所以就會很容易找到文件,并輸出: ``` Lib static function test App\Android static function test App\Ios static function test ``` 好了。基本自動加載的東西就講完了。很實用的東西。 ## **同命名空間下的相互調用** 在平時我們使用命令空間時,有時候可能是在同一個命名空間下的2個類文件在相互調用。這個時候就要注意,在自動調用的問題了。 比如Lib\\Factory.php 和 Lib\\Db\\MySQL.php 我想在 Lib\\Factory.php 中調用 Lib\\Db\\MySQL.php。怎么調用呢?以下是錯誤的示范: ``` newLib\Db\MySQL(); //報錯,提示說 D:\wamp\www\testphp\module\Lib\Lib\Db\MySQL.php is not exist ``` newLib\Db\MySQL(); //報錯,提示說 D:\wamp\www\testphp\module\Lib\Lib\Db\MySQL.php is not exist ``` newMySQL();//直接這樣就可以了。 newDb\MySQL();//如果有個Db文件夾,就這樣。 ``` 還有一種方法就是使用 use 。使用user就可以帶上Lib了。use使用的是絕對路徑。 ``` useLib\Db\MySQL; newMySQL(); ``` 我想在 Lib\Db\MySQL.php 中調用 Lib\Register.php。怎么調用呢? 應該這樣 ``` useLib\Register; Register::getInstance(); ``` 因為現在已經在Lib\Db這樣一個命名空間了,如果你不用use,而是使用Lib\Register::getInstance()或者使用Register::getInstance()的話。將是在Lib\Db這個空間下進行相對路徑的加載,是錯誤的。
                  <ruby id="bdb3f"></ruby>

                  <p id="bdb3f"><cite id="bdb3f"></cite></p>

                    <p id="bdb3f"><cite id="bdb3f"><th id="bdb3f"></th></cite></p><p id="bdb3f"></p>
                      <p id="bdb3f"><cite id="bdb3f"></cite></p>

                        <pre id="bdb3f"></pre>
                        <pre id="bdb3f"><del id="bdb3f"><thead id="bdb3f"></thead></del></pre>

                        <ruby id="bdb3f"><mark id="bdb3f"></mark></ruby><ruby id="bdb3f"></ruby>
                        <pre id="bdb3f"><pre id="bdb3f"><mark id="bdb3f"></mark></pre></pre><output id="bdb3f"></output><p id="bdb3f"></p><p id="bdb3f"></p>

                        <pre id="bdb3f"><del id="bdb3f"><progress id="bdb3f"></progress></del></pre>

                              <ruby id="bdb3f"></ruby>

                              哎呀哎呀视频在线观看