国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 語言 > PHP > 正文

PHP中的自動加載操作實現方法詳解

2024-05-05 00:10:46
字體:
來源:轉載
供稿:網友

本文實例講述了PHP中的自動加載操作實現方法。分享給大家供大家參考,具體如下:

what is 自動加載?

或許你已經對自動加載有所了解。簡單描述一下:自動加載就是我們在new一個class的時候,不需要手動去寫require來導入這個class.php文件,程序自動幫我們加載導入進來。這是php5.1.2(好像是)版本新加入一個功能,他解放了程序員的雙手,不需要手動寫那么多的require,變得有那么點智能的感覺。

自動加載可以說是現代PHP框架的根基,任何牛逼的框架或者架構都會用到它,它發明出來的理由是啥呢?一個字:懶。因為項目越來愈大,相關聯的類庫文件越來越多,我們不可能再像小項目那樣在一個文件中全部手動一個一個require

如何才能自動加載呢? PHP 5.2版本更新了自動加載需要的一個魔術方法——__autoload($class_name)

正是這個神奇的內置魔術函數,才能讓我們這些屌絲偷懶。我們來看下這個如何使用它。

1. 自動加載的原理以及__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()這個方法再看看:

$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,比如:

<?phpfunction __autoload($className){  require $className .'.php';}$db =new DB();$info =newInfo();$gender =newGender();$name =newName();//也是支持靜態方法直接調用的Height::test();

2. 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就好了。

3. 多個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)

4. spl_autoload_register自動加載+namespace命名空間 的使用

前面已經說過,自動加載現在是PHP現代框架的基石,基本都是spl_autoload_register來實現自動加載。namespace也是使用比較多的。所以spl_autoload_register + namespace 就成為了一個主流。根據PSR-0的規范,namespace命名已經非常規范化,所以用namespace就能找到詳細的路徑,從而找到類文件。

我們舉例子來看下:

AutoLoading/loading

<?phpnamespaceAutoLoading;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的方式調用,注意:不能使用usespl_autoload_register("//AutoLoading//loading::autoload");// 調用三個namespace類//定位到Lib目錄下的Name.phpLib/Name::test();//定位到App目錄下Android目錄下的Name.phpApp/Android/Name::test();//定位到App目錄下Ios目錄下的Name.phpApp/Ios/Name::test();

由于我們是采用PSR-O方式來定義namespace的命名的,所以很好的定位到這個文件的在哪個目錄下了。很爽。對不對。

APP/Android/Name

<?phpnamespaceApp/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

好了。基本自動加載的東西就講完了。很實用的東西。

4. 同命名空間下的相互調用

在平時我們使用命令空間時,有時候可能是在同一個命名空間下的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

看到沒?這種方式是在Lib/命名空間的基礎上來加載的。所以會加載2個Lib。這種方式相當于相對路徑在加載。

正確的做法是,如果是在同一個命名空間下平級的2個文件。可以直接調用,不用命名空間。

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這個空間下進行相對路徑的加載,是錯誤的。

希望本文所述對大家PHP程序設計有所幫助。


注:相關教程知識閱讀請移步到PHP教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表

圖片精選

主站蜘蛛池模板: 儋州市| 铁岭县| 鞍山市| 上虞市| 穆棱市| 彭阳县| 沈丘县| 盘山县| 柳林县| 旌德县| 义马市| 绍兴市| 揭西县| 剑阁县| 宣汉县| 临高县| 镇江市| 夏津县| 海门市| 平塘县| 哈尔滨市| 峨边| 扬中市| 新安县| 中西区| 兴安盟| 武穴市| 濮阳县| 洪江市| 芒康县| 望谟县| 河源市| 都匀市| 西华县| 深水埗区| 五华县| 德兴市| 连州市| 迭部县| 汶川县| 南木林县|