本文實(shí)例講述了PHP中的自動加載操作實(shí)現(xiàn)方法。分享給大家供大家參考,具體如下:
what is 自動加載?
或許你已經(jīng)對自動加載有所了解。簡單描述一下:自動加載就是我們在new一個class的時候,不需要手動去寫require來導(dǎo)入這個class.php文件,程序自動幫我們加載導(dǎo)入進(jìn)來。這是php5.1.2(好像是)版本新加入一個功能,他解放了程序員的雙手,不需要手動寫那么多的require,變得有那么點(diǎn)智能的感覺。
自動加載可以說是現(xiàn)代PHP框架的根基,任何牛逼的框架或者架構(gòu)都會用到它,它發(fā)明出來的理由是啥呢?一個字:懶。因?yàn)轫?xiàng)目越來愈大,相關(guān)聯(lián)的類庫文件越來越多,我們不可能再像小項(xiàng)目那樣在一個文件中全部手動一個一個require
。
如何才能自動加載呢? PHP 5.2版本更新了自動加載需要的一個魔術(shù)方法——__autoload($class_name)
正是這個神奇的內(nèi)置魔術(shù)函數(shù),才能讓我們這些屌絲偷懶。我們來看下這個如何使用它。
1. 自動加載的原理以及__autoload的使用
自動加載的原理,就是在我們new
一個class的時候,PHP系統(tǒng)如果找不到你這個類,就會去自動調(diào)用本文件中的__autoload($class_name)
方法,我們new
的這個class_name 就成為這個方法的參數(shù)。所以我們就可以在這個方法中根據(jù)我們需要new class_name的各種判斷和劃分就去require對應(yīng)的路徑類文件,從而實(shí)現(xiàn)自動加載。
我們先一步步來,看下__autoload()
的自動調(diào)用,看個例子:
index.php
如果我們不手動導(dǎo)入DB類,程序可能會報(bào)錯,說找不到這個類:
Fatal error: Class 'DB' not found in D:\wamp\www\testphp\autoload\index.php on line 3
那么,我們現(xiàn)在加入__autoload()
這個方法再看看:
$db =new DB();
function __autoload($className)
{
echo $className;
exit();
}
根據(jù)上面自動加載機(jī)制的描述,你分析下會輸出什么? 沒錯:肯定是輸出:DB, 也就是我們需要new
的類的類名。所以,這個時候我們就可以在__autoload()
方法里,根據(jù)需要去加載類庫文件了。
index.php
$db =new DB();
function __autoload($className)
{
require $className .'.php';
}
DB.php
class DB
{
publicfunction __construct()
{
echo 'Hello DB';
}
}
這樣子我們就很輕松的將我們需要new
的class 全部導(dǎo)入了進(jìn)來,這樣子,我們就可以輕松的new
N個class,比如:
?php
function __autoload($className)
{
require $className .'.php';
}
$db =new DB();
$info =newInfo();
$gender =newGender();
$name =newName();
//也是支持靜態(tài)方法直接調(diào)用的
Height::test();
2. spl_autoload_register的使用
小的項(xiàng)目,用__autoload()
就能實(shí)現(xiàn)基本的自動加載了。但是如果一個項(xiàng)目過大,或者需要不同的自動加載來加載不同路徑的文件,這個時候__autoload
就悲劇了,原因是一個項(xiàng)目中僅能有一個這樣的 __autoload()
函數(shù),因?yàn)?PHP 不允許函數(shù)重名,也就是說你不能聲明2個__autoload()
函數(shù)文件,否則會報(bào)致命錯誤,我了個大擦,那怎么辦呢?放心,你想到的,PHP開發(fā)大神早已經(jīng)想到。
所以spl_autoload_register()
這樣又一個牛逼函數(shù)誕生了,并且取而代之它。它執(zhí)行效率更高,更靈活
先看下它如何使用吧:
當(dāng)我們?nèi)?code>new一個找不到的class時,PHP就會去自動調(diào)用sql_autoload_resister
注冊的函數(shù),這個函數(shù)通過它的參數(shù)傳進(jìn)去:
sql_autoload_resister($param) 這個參數(shù)可以有多種形式:
sql_autoload_resister('load_function'); //函數(shù)名
sql_autoload_resister(array('load_object', 'load_function')); //類和靜態(tài)方法
sql_autoload_resister('load_object::load_function'); //類和方法的靜態(tài)調(diào)用
//php 5.3之后,也可以像這樣支持匿名函數(shù)了。
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函數(shù)注冊到自動加載隊(duì)列中。
$db =new DB();//找不到DB類,就會自動去調(diào)用剛注冊的load1函數(shù)了
上面就是實(shí)現(xiàn)了自動加載的方式,我們同樣也可以用類加載的方式調(diào)用,但是必須是static方法:
class autoloading {
//必須是靜態(tài)方法,不然報(bào)錯
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
是可以多次重復(fù)使用的,這一點(diǎn)正是解決了__autoload
的短板,那么如果一個頁面有多個,執(zhí)行順序是按照注冊的順序,一個一個往下找,如果找到了就停止。
我們來看下這個例子,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個自動加載函數(shù)。執(zhí)行結(jié)果是啥呢?
1Hello DB
123Hello Info
我們分析下:
new DB
的時候,就按照注冊順序,先去找load1()函數(shù)了,發(fā)現(xiàn)找到了,就停止了,所以輸出1 Hello Word
new Info
的時候,先是安裝注冊順序,先找load1()
, 所以輸出了1,發(fā)現(xiàn)沒找到,就去load2()
里面去找,所以輸出了2,還是沒這個文件,就去__autoload()
函數(shù)里找,所以,先輸出了3,再輸出Hello Info
注意,前面說過,spl_autoload_register
使用時,__autoload
會無效,有時候,我們希望它繼續(xù)有效,就可以也將它注冊進(jìn)來,就可以繼續(xù)使用。
我們可以打印spl_autoload_functions()
函數(shù),來顯示一共注冊了多少個自動加載:
var_dump(spl_autoload_functions());
//數(shù)組的形式輸出
array (size=3)
0 => string 'load1' (length=5)
1 => string 'load2' (length=5)
2 => string '__autoload' (length=10)
4. spl_autoload_register自動加載+namespace命名空間 的使用
前面已經(jīng)說過,自動加載現(xiàn)在是PHP現(xiàn)代框架的基石,基本都是spl_autoload_register
來實(shí)現(xiàn)自動加載。namespace
也是使用比較多的。所以spl_autoload_register
+ namespace
就成為了一個主流。根據(jù)PSR-0的規(guī)范,namespace
命名已經(jīng)非常規(guī)范化,所以用namespace
就能找到詳細(xì)的路徑,從而找到類文件。
我們舉例子來看下:
AutoLoading\loading
?php
namespaceAutoLoading;
class loading {
public static function autoload($className)
{
//根據(jù)PSR-O的第4點(diǎn) 把 \ 轉(zhuǎn)換層(目錄風(fēng)格符) DIRECTORY_SEPARATOR ,
//便于兼容Linux文件找。Windows 下(/ 和 \)是通用的
//由于namspace 很規(guī)格,所以直接很快就能找到
$fileName = str_replace('\\', DIRECTORY_SEPARATOR, DIR .'\\'. $className).'.php';
if(is_file($fileName)){
require $fileName;
}else{
echo $fileName .' is not exist';die;
}
}
}
上面就是一個自動加載的核心思想方法。下面我們就來spl_autoload_register
來注冊這個函數(shù):
index.php
?php
//定義當(dāng)前的目錄絕對路徑
define('DIR', dirname(__FILE__));
//加載這個文件
require DIR .'/loading.php';
//采用`命名空間`的方式注冊。php 5.3 加入的
//也必須是得是static靜態(tài)方法調(diào)用,然后就像加載namespace的方式調(diào)用,注意:不能使用use
spl_autoload_register("\\AutoLoading\\loading::autoload");
// 調(diào)用三個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
好了?;咀詣蛹虞d的東西就講完了。很實(shí)用的東西。
4. 同命名空間下的相互調(diào)用
在平時我們使用命令空間時,有時候可能是在同一個命名空間下的2個類文件在相互調(diào)用。這個時候就要注意,在自動調(diào)用的問題了。
比如Lib\Factory.php 和 Lib\Db\MySQL.php
我想在 Lib\Factory.php 中調(diào)用 Lib\Db\MySQL.php。怎么調(diào)用呢?以下是錯誤的示范:
newLib\Db\MySQL();
//報(bào)錯,提示說 D:\wamp\www\testphp\module\Lib\Lib\Db\MySQL.php is not exist
看到?jīng)]?這種方式是在Lib\命名空間的基礎(chǔ)上來加載的。所以會加載2個Lib。這種方式相當(dāng)于相對路徑在加載。
正確的做法是,如果是在同一個命名空間下平級的2個文件??梢灾苯诱{(diào)用,不用命名空間。
newMySQL();//直接這樣就可以了。
newDb\MySQL();//如果有個Db文件夾,就這樣。
還有一種方法就是使用 use 。使用user就可以帶上Lib了。use使用的是絕對路徑。
useLib\Db\MySQL;
newMySQL();
我想在 Lib\Db\MySQL.php 中調(diào)用 Lib\Register.php。怎么調(diào)用呢?
應(yīng)該這樣
useLib\Register;
Register::getInstance();
因?yàn)楝F(xiàn)在已經(jīng)在Lib\Db這樣一個命名空間了,如果你不用use
,而是使用Lib\Register::getInstance()
或者使用Register::getInstance()
的話。將是在Lib\Db這個空間下進(jìn)行相對路徑的加載,是錯誤的。
更多關(guān)于PHP相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《php面向?qū)ο蟪绦蛟O(shè)計(jì)入門教程》、《PHP數(shù)組(Array)操作技巧大全》、《PHP基本語法入門教程》、《PHP運(yùn)算與運(yùn)算符用法總結(jié)》、《php字符串(string)用法總結(jié)》、《php+mysql數(shù)據(jù)庫操作入門教程》及《php常見數(shù)據(jù)庫操作技巧匯總》
希望本文所述對大家PHP程序設(shè)計(jì)有所幫助。
您可能感興趣的文章:- 說說PHP的autoLoad自動加載機(jī)制
- php自動加載的兩種實(shí)現(xiàn)方法
- PHP管理依賴(dependency)關(guān)系工具 Composer的自動加載(autoload)
- PHP命名空間和自動加載類
- PHP的autoload自動加載機(jī)制使用說明
- php自動加載方式集合
- PHP spl_autoload_register實(shí)現(xiàn)自動加載研究
- PHP動態(tài)地創(chuàng)建屬性和方法, 對象的復(fù)制, 對象的比較,加載指定的文件,自動加載類文件,命名空間
- PHP中類的自動加載的方法
- PHP autoload與spl_autoload自動加載機(jī)制的深入理解
- PHP 自動加載的簡單實(shí)現(xiàn)(推薦)