溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

使用PHP怎么實(shí)現(xiàn)一個(gè)自動(dòng)加載操作功能

發(fā)布時(shí)間:2021-02-08 15:22:40 來(lái)源:億速云 閱讀:159 作者:Leah 欄目:開發(fā)技術(shù)

本篇文章給大家分享的是有關(guān)使用PHP怎么實(shí)現(xiàn)一個(gè)自動(dòng)加載操作功能,小編覺得挺實(shí)用的,因此分享給大家學(xué)習(xí),希望大家閱讀完這篇文章后可以有所收獲,話不多說(shuō),跟著小編一起來(lái)看看吧。

1. 自動(dòng)加載的原理以及__autoload的使用

自動(dòng)加載的原理,就是在我們new一個(gè)class的時(shí)候,PHP系統(tǒng)如果找不到你這個(gè)類,就會(huì)去自動(dòng)調(diào)用本文件中的__autoload($class_name)方法,我們new的這個(gè)class_name 就成為這個(gè)方法的參數(shù)。所以我們就可以在這個(gè)方法中根據(jù)我們需要new class_name的各種判斷和劃分就去require對(duì)應(yīng)的路徑類文件,從而實(shí)現(xiàn)自動(dòng)加載。

我們先一步步來(lái),看下__autoload()的自動(dòng)調(diào)用,看個(gè)例子:

index.php

$db =new DB();

如果我們不手動(dòng)導(dǎo)入DB類,程序可能會(huì)報(bào)錯(cuò),說(shuō)找不到這個(gè)類:

Fatal error: Class 'DB' not found in D:\wamp\www\testphp\autoload\index.php on line 3

那么,我們現(xiàn)在加入__autoload()這個(gè)方法再看看:

$db =new DB();
function __autoload($className)
{
  echo $className;
  exit();
}

根據(jù)上面自動(dòng)加載機(jī)制的描述,你分析下會(huì)輸出什么? 沒錯(cuò):肯定是輸出:DB, 也就是我們需要new 的類的類名。所以,這個(gè)時(shí)候我們就可以在__autoload()方法里,根據(jù)需要去加載類庫(kù)文件了。

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)來(lái),這樣子,我們就可以輕松的new N個(gè)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)基本的自動(dòng)加載了。但是如果一個(gè)項(xiàng)目過(guò)大,或者需要不同的自動(dòng)加載來(lái)加載不同路徑的文件,這個(gè)時(shí)候__autoload就悲劇了,原因是一個(gè)項(xiàng)目中僅能有一個(gè)這樣的 __autoload() 函數(shù),因?yàn)?PHP 不允許函數(shù)重名,也就是說(shuō)你不能聲明2個(gè)__autoload()函數(shù)文件,否則會(huì)報(bào)致命錯(cuò)誤,我了個(gè)大擦,那怎么辦呢?放心,你想到的,PHP開發(fā)大神早已經(jīng)想到。

所以spl_autoload_register()這樣又一個(gè)牛逼函數(shù)誕生了,并且取而代之它。它執(zhí)行效率更高,更靈活

先看下它如何使用吧:

當(dāng)我們?nèi)?code>new一個(gè)找不到的class時(shí),PHP就會(huì)去自動(dòng)調(diào)用sql_autoload_resister注冊(cè)的函數(shù),這個(gè)函數(shù)通過(guò)它的參數(shù)傳進(jìn)去:

sql_autoload_resister($param) 這個(gè)參數(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ù)注冊(cè)到自動(dòng)加載隊(duì)列中。
$db =new DB();//找不到DB類,就會(huì)自動(dòng)去調(diào)用剛注冊(cè)的load1函數(shù)了

上面就是實(shí)現(xiàn)了自動(dòng)加載的方式,我們同樣也可以用類加載的方式調(diào)用,但是必須是static方法:

class autoloading {
//必須是靜態(tài)方法,不然報(bào)錯(cuò)
  public static function load($className)
  {
    require $className .'.php';
  }
}
//2種方法都可以
spl_autoload_register(array('autoloading','load'));
spl_autoload_register('autoloading::load');
$db =new DB();//會(huì)自動(dòng)找到

需要注意的是,如果你同時(shí)使用spl_autoload_register__autoload,__autoload會(huì)失效!??! 再說(shuō)了,本來(lái)就是替換它的,就一心使用spl_autoload_register就好了。

3. 多個(gè)spl_autoload_register的使用

spl_autoload_register是可以多次重復(fù)使用的,這一點(diǎn)正是解決了__autoload的短板,那么如果一個(gè)頁(yè)面有多個(gè),執(zhí)行順序是按照注冊(cè)的順序,一個(gè)一個(gè)往下找,如果找到了就停止。

我們來(lái)看下這個(gè)例子,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';
  }
}
//注冊(cè)了3個(gè)
spl_autoload_register('load1');
spl_autoload_register('load2');
spl_autoload_register('__autoload');
$db =new DB();//DB就在本目錄下
$info =newInfo();//Info 在/lib/Info.php

我們注冊(cè)了3個(gè)自動(dòng)加載函數(shù)。執(zhí)行結(jié)果是啥呢?

1Hello DB
123Hello Info

我們分析下:

new DB的時(shí)候,就按照注冊(cè)順序,先去找load1()函數(shù)了,發(fā)現(xiàn)找到了,就停止了,所以輸出1 Hello Word

new Info的時(shí)候,先是安裝注冊(cè)順序,先找load1(), 所以輸出了1,發(fā)現(xiàn)沒找到,就去load2()里面去找,所以輸出了2,還是沒這個(gè)文件,就去__autoload()函數(shù)里找,所以,先輸出了3,再輸出Hello Info

注意,前面說(shuō)過(guò),spl_autoload_register使用時(shí),__autoload會(huì)無(wú)效,有時(shí)候,我們希望它繼續(xù)有效,就可以也將它注冊(cè)進(jìn)來(lái),就可以繼續(xù)使用。

我們可以打印spl_autoload_functions()函數(shù),來(lái)顯示一共注冊(cè)了多少個(gè)自動(dòng)加載:

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自動(dòng)加載+namespace命名空間 的使用

前面已經(jīng)說(shuō)過(guò),自動(dòng)加載現(xiàn)在是PHP現(xiàn)代框架的基石,基本都是spl_autoload_register來(lái)實(shí)現(xiàn)自動(dòng)加載。namespace也是使用比較多的。所以spl_autoload_register + namespace 就成為了一個(gè)主流。根據(jù)PSR-0的規(guī)范,namespace命名已經(jīng)非常規(guī)范化,所以用namespace就能找到詳細(xì)的路徑,從而找到類文件。

我們舉例子來(lái)看下:

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;
    }
  }
}

上面就是一個(gè)自動(dòng)加載的核心思想方法。下面我們就來(lái)spl_autoload_register來(lái)注冊(cè)這個(gè)函數(shù):

index.php

<?php
//定義當(dāng)前的目錄絕對(duì)路徑
define('DIR', dirname(__FILE__));
//加載這個(gè)文件
require DIR .'/loading.php';
//采用`命名空間`的方式注冊(cè)。php 5.3 加入的
//也必須是得是static靜態(tài)方法調(diào)用,然后就像加載namespace的方式調(diào)用,注意:不能使用use
spl_autoload_register("\\AutoLoading\\loading::autoload");
// 調(diào)用三個(gè)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方式來(lái)定義namespace的命名的,所以很好的定位到這個(gè)文件的在哪個(gè)目錄下了。很爽。對(duì)不對(duì)。

APP\Android\Name

<?php
namespaceApp\Android;
className
{
  public function __construct()
  {
    echo __NAMESPACE__ ."<br>";
  }
  public static function test()
  {
    echo __NAMESPACE__ .' static function test <br>';
  }
}

所以就會(huì)很容易找到文件,并輸出:

Lib static function test
App\Android static function test
App\Ios static function test

好了。基本自動(dòng)加載的東西就講完了。很實(shí)用的東西。

4. 同命名空間下的相互調(diào)用

在平時(shí)我們使用命令空間時(shí),有時(shí)候可能是在同一個(gè)命名空間下的2個(gè)類文件在相互調(diào)用。這個(gè)時(shí)候就要注意,在自動(dòng)調(diào)用的問(wèn)題了。

比如Lib\Factory.php 和 Lib\Db\MySQL.php

我想在 Lib\Factory.php 中調(diào)用 Lib\Db\MySQL.php。怎么調(diào)用呢?以下是錯(cuò)誤的示范:

newLib\Db\MySQL();
//報(bào)錯(cuò),提示說(shuō) D:\wamp\www\testphp\module\Lib\Lib\Db\MySQL.php is not exist

看到?jīng)]?這種方式是在Lib\命名空間的基礎(chǔ)上來(lái)加載的。所以會(huì)加載2個(gè)Lib。這種方式相當(dāng)于相對(duì)路徑在加載。

正確的做法是,如果是在同一個(gè)命名空間下平級(jí)的2個(gè)文件??梢灾苯诱{(diào)用,不用命名空間。

newMySQL();//直接這樣就可以了。
newDb\MySQL();//如果有個(gè)Db文件夾,就這樣。

還有一種方法就是使用 use 。使用user就可以帶上Lib了。use使用的是絕對(duì)路徑。

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這樣一個(gè)命名空間了,如果你不用use,而是使用Lib\Register::getInstance()或者使用Register::getInstance()的話。將是在Lib\Db這個(gè)空間下進(jìn)行相對(duì)路徑的加載,是錯(cuò)誤的。

以上就是使用PHP怎么實(shí)現(xiàn)一個(gè)自動(dòng)加載操作功能,小編相信有部分知識(shí)點(diǎn)可能是我們?nèi)粘9ぷ鲿?huì)見到或用到的。希望你能通過(guò)這篇文章學(xué)到更多知識(shí)。更多詳情敬請(qǐng)關(guān)注億速云行業(yè)資訊頻道。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

php
AI