亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

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

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

thinkphp5.1框架中如何實現容器和門面

發布時間:2021-08-04 13:51:26 來源:億速云 閱讀:147 作者:小新 欄目:開發技術

這篇文章主要介紹了thinkphp5.1框架中如何實現容器和門面,具有一定借鑒價值,感興趣的朋友可以參考下,希望大家閱讀完這篇文章之后大有收獲,下面讓小編帶著大家一起了解一下。

具體如下:

tp5.1中引入了容器(Container)和門面(Facade)這兩個新的類

官方文檔已經給出了定義:

容器(Container)實現類的統一管理,確保對象實例的唯一性。

門面(Facade)為容器(Container)中的類提供了一個靜態調用接口,相比于傳統的靜態方法調用, 帶來了更好的可測試性和擴展性,你可以為任何的非靜態類庫定義一個facade類。

深入源碼,我們來看看它到底是如何實現的:

// 在框架目錄下的base.php文件
// 注冊核心類到容器
Container::getInstance()->bind([
  'app'          => App::class,
  'build'         => Build::class,
  'cache'         => Cache::class,
  'config'        => Config::class,
  ...
]);
// 注冊核心類的靜態代理
Facade::bind([
  facade\App::class   => App::class,
  facade\Build::class  => Build::class,
  facade\Cache::class  => Cache::class,
  facade\Config::class  => Config::class,
  ...
]);
// 注冊類庫別名
Loader::addClassAlias([
  'App'   => facade\App::class,
  'Build'  => facade\Build::class,
  'Cache'  => facade\Cache::class,
  'Config'  => facade\Config::class,
  ...
]);

容器實現:

這里,框架已經幫我們綁定了系統常用類到容器中,在之后使用時,只需要調用助手函數 app()進行容器中的類解析調用,對于已經綁定的類標識,會自動快速實例化。

// 實例化緩存類
app('cache');
// app('cache', ['file']); 參數化調用
// 相當于執行了
Container::get('cache');
// 查看源碼,Container調用的其實是make方法,在該方法里調用反射等實現類的實例化,過程如下:
public function make($abstract, $vars = [], $newInstance = false)
{
  if (true === $vars) {
    // 總是創建新的實例化對象
    $newInstance = true;
    $vars    = [];
  }
  if (isset($this->instances[$abstract]) && !$newInstance) {
    $object = $this->instances[$abstract];
  } else {
    if (isset($this->bind[$abstract])) {
      $concrete = $this->bind[$abstract];
       // 閉包實現
      if ($concrete instanceof \Closure) {
        $object = $this->invokeFunction($concrete, $vars);
      } else {
        $object = $this->make($concrete, $vars, $newInstance);
      }
    } else {
       // 反射實現
      $object = $this->invokeClass($abstract, $vars);
    }
    if (!$newInstance) {
      $this->instances[$abstract] = $object;
    }
  }
  return $object;
}
/**
 * 調用反射執行類的實例化 支持依賴注入
 * @access public
 * @param string  $class 類名
 * @param array   $vars 變量
 * @return mixed
 */
public function invokeClass($class, $vars = [])
{
  $reflect   = new \ReflectionClass($class);
  $constructor = $reflect->getConstructor();
  if ($constructor) {
    $args = $this->bindParams($constructor, $vars);
  } else {
    $args = [];
  }
  return $reflect->newInstanceArgs($args);
}
/**
 * 執行函數或者閉包方法 支持參數調用
 * @access public
 * @param string|array|\Closure $function 函數或者閉包
 * @param array         $vars   變量
 * @return mixed
 */
public function invokeFunction($function, $vars = [])
{
  $reflect = new \ReflectionFunction($function);
  $args  = $this->bindParams($reflect, $vars);
  return $reflect->invokeArgs($args);
}

簡而言之,容器內部是通過反射類或閉包等來實現類的實例化。

門面實現:

以一個例子來分析:

facade\Config::get('app_debug');

我們來分析一下它的實現方式:

// thinkphp\library\facade\Config 類
namespace think\facade;
use think\Facade;
class Config extends Facade
{
}
// 從源代碼上看 Config本身沒有任何方法,它繼承了Facade的方法,但Facade并沒有get這個靜態方法
// 此時,系統自動觸發了魔術方法:__callStatic(),Facade重寫了此方法:
public static function __callStatic($method, $params)
{
  return call_user_func_array([static::createFacade(), $method], $params);
}
// 可見,最后調用的是用戶自定義函數:call_user_func_array([實例, 方法], 參數),為了獲得Config實例,Facade又定義了一個獲取對象的方法:
/**
 * 創建Facade實例
 * @static
 * @access protected
 * @param string  $class     類名或標識
 * @param array   $args      變量
 * @param bool   $newInstance  是否每次創建新的實例
 * @return object
 */
protected static function createFacade($class = '', $args = [], $newInstance = false)
{
  $class    = $class ?: static::class;
  $facadeClass = static::getFacadeClass();
  if ($facadeClass) {
    $class = $facadeClass;
  } elseif (isset(self::$bind[$class])) {
    $class = self::$bind[$class];
  }
  if (static::$alwaysNewInstance) {
    $newInstance = true;
  }
  return Container::getInstance()->make($class, $args, $newInstance);
}
// 其內部是通過容器來實例化對象
// 因為在base.php中已經將 think\Config 類綁定到 config 這個標識
Container::getInstance()->bind([
  'config' => Config::class
])
// 在 createFacade 方法中,獲取類的名稱:$class = $class ?: static::class; 即得到 config 這個標識
// 在容器的make方法中,根據config標識,找到綁定的 think\Config 類,并調用其動態方法 get。
facade\Config::get('app_debug');
// 最后調用的是:
(new think\Config())->get('app_debug');

簡而言之,門面的實現是通過PHP的魔術方法 __callStatic,再配合容器來實現動態類的靜態化調用。

感謝你能夠認真閱讀完這篇文章,希望小編分享的“thinkphp5.1框架中如何實現容器和門面”這篇文章對大家有幫助,同時也希望大家多多支持億速云,關注億速云行業資訊頻道,更多相關知識等著你來學習!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

兴宁市| 乐东| 闽清县| 湟中县| 荣昌县| 巢湖市| 延庆县| 百色市| 突泉县| 南雄市| 霍州市| 青岛市| 定西市| 山西省| 孟州市| 河源市| 米易县| 新营市| 乌兰察布市| 郯城县| 济源市| 永顺县| 义马市| 泰宁县| 满洲里市| 准格尔旗| 贵南县| 德阳市| 依安县| 台东市| 陆丰市| 深州市| 沅陵县| 和顺县| 康定县| 淮北市| 双柏县| 常熟市| 遵义县| 连山| 宿松县|