好湿?好紧?好多水好爽自慰,久久久噜久噜久久综合,成人做爰A片免费看黄冈,机机对机机30分钟无遮挡

主頁 > 知識庫 > thinkphp5.1框架中容器(Container)和門面(Facade)的實現方法分析

thinkphp5.1框架中容器(Container)和門面(Facade)的實現方法分析

熱門標簽:仁和怎么申請400開頭的電話 廣東地市地圖標注 高德地圖標注家 廣州防封卡外呼系統多少錢一個月 外呼系統撥打暫時無法接通 長春人工外呼系統服務商 怎么向銷售公司推銷外呼系統 哪里辦理400電話 江西手機自動外呼防封系統是什么

本文實例講述了thinkphp5.1框架中容器(Container)和門面(Facade)的實現方法。分享給大家供大家參考,具體如下:

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,再配合容器來實現動態類的靜態化調用。

更多關于thinkPHP相關內容感興趣的讀者可查看本站專題:《ThinkPHP入門教程》、《thinkPHP模板操作技巧總結》、《ThinkPHP常用方法總結》、《codeigniter入門教程》、《CI(CodeIgniter)框架進階教程》、《Zend FrameWork框架入門教程》及《PHP模板技術總結》。

希望本文所述對大家基于ThinkPHP框架的PHP程序設計有所幫助。

您可能感興趣的文章:
  • PHP設計模式(九)外觀模式Facade實例詳解【結構型】
  • PHP設計模式之外觀模式(Facade)入門與應用詳解
  • 詳解PHP中的外觀模式facade pattern
  • 學習php設計模式 php實現門面模式(Facade)
  • php設計模式 Facade(外觀模式)
  • PHP設計模式之PHP迭代器模式講解
  • PHP設計模式之工廠模式(Factory Pattern)的講解
  • 淺談PHP設計模式之門面模式Facade

標簽:黔東 湘西 梅河口 惠州 廈門 文山 海北 濮陽

巨人網絡通訊聲明:本文標題《thinkphp5.1框架中容器(Container)和門面(Facade)的實現方法分析》,本文關鍵詞  thinkphp5.1,框架,中,容器,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《thinkphp5.1框架中容器(Container)和門面(Facade)的實現方法分析》相關的同類信息!
  • 本頁收集關于thinkphp5.1框架中容器(Container)和門面(Facade)的實現方法分析的相關信息資訊供網民參考!
  • 推薦文章
    主站蜘蛛池模板: 女仆主动张开双腿给主人调教| 91精品夜夜夜一区二区不卡牛牛| 我们的2018在线看免费| 动漫美女的隐私免费视频| 她是第三种绝色| 扒开她的小说让我?吮| 东北老熟女高潮45分钟视频 | 小黄文在线| 中国zzzwww大片免费视频| heyzo无码亚洲中文字幕| 18岁女人毛片| 久久91精品国产91久久小草 | 成人无码精品一区二区黑寡妇在线| 国产一级婬乱片A片A片片| 香蕉视频禁止18| 日本生活中的玛丽| 蜜桃AV鲁一鲁一鲁一鲁俄罗斯的| EEUSS秋霞成人影院| 污污软件大全| 性激烈的香港三级视频| 玩弄我的三位美艳馊TXT小说下载| 噜噜狠狠色综合久色AⅤ一区| 肉文男男| 毛片黄| 91亚洲精品久久久久久久久久久久| 饥渴丰满的少妇喷潮抽搐A片| 欧洲欧美人成免费观看| 91色国产在线| ?国产精品嫩草影院AV蜜芽| 双性受在寝室被c翻双龙H| 三上悠亚ssni在线| 印度毛茸茸bbbbb| 爱爱动图好硬好湿好爽| 免费?无码?国产AV在线观看| 久久人人超碰| 看性过程三级视频在线观看| 妞干网视频| 小师弟只想咸鱼| 曰本毛片Va看到爽| 97国产大学生情侣酒店| 26uuu另类亚洲欧美日本一|