[TOC]
# 容器類
~~~
<?php
class Container
{
//用于提供實例的回調函數,真正的容器還會裝實例等其他內容.從而實現單列等高級功能
public $binding = [];
//綁定接口和生成相應實例的回調函數
public function bind($abstract, $concrete = null, $shared = false)
{
if (!$concrete instanceof Closure) {
//如果提供的參數不是回調函數則產生默認的回調函數
$concrete = $this->getClosure($abstract, $concrete);
}
$this->binding[$abstract] = compact('concrete', 'shared');
}
//默認生成的實例的回調函數
protected function getClosure($abstract, $concrete)
{
//生成實例的回調函數,$c一般為IOC容器對象,在調用回調生成實例時提供
//即build函數中的$concrete($this)
return function ($c) use ($abstract, $concrete) {
$method = ($abstract == $concrete) ? 'build' : 'make';
//調用的是容器的build或make方法生成的實例
return $c->$method($concrete);
};
}
//生成實例對象,首先要解決接口和要實例化類之間的依賴關系
public function make($abstract)
{
$concrete = $this->getConcrete($abstract);
if ($this->isBuildable($concrete, $abstract)) {
$object = $this->build($concrete);
} else {
$object = $this->make($concrete);
}
return $object;
}
public function isBuildable($concrete, $abstract)
{
return $concrete === $abstract || $concrete instanceof Closure;
}
//獲取綁定的回調函數
protected function getConcrete($abstract)
{
if (!isset($this->binding[$abstract])) {
return $abstract;
}
return $this->binding[$abstract]['concrete'];
}
//實例化對象
public function build($concrete) {
if($concrete instanceof Closure) {
return $concrete($this);
}
//反射...
$reflector = new ReflectionClass($concrete);
if(!$reflector->isInstantiable()) {
echo $message = "Target [$concrete] is not instantiable";
}
//獲取要實例化對象的構造函數
$constructor = $reflector->getConstructor();
//沒有定義構造函數,只有默認的構造函數,說明構造函數參數個數為空
if(is_null($constructor)) {
return new $concrete;
}
//獲取構造函數所需要的所有參數
$dependencies = $constructor->getParameters();
$instances = $this->getDependencies($dependencies);
//從給出的數組參數在中實例化對象
return $reflector->newInstanceArgs($instances);
}
/**
* 獲取構建類所需要的所有依賴,級構造函數所需要的參數 ,
*/
protected function getDependencies($paramters) {
$dependencies = [];
foreach ($paramters as $paramter) {
//獲取到參數名稱.
$dep = $paramter->getClass();
if(is_null($dep)){
$dependencies = null;
}else{
$dependencies[] = $this->resolveClass($paramter);
}
}
return (array)$dependencies;
}
/**
* 實例化 構造函數中所需要的參數.
*/
protected function resolveClass(ReflectionParameter $parameter) {
$name = $parameter->getClass()->name;
return $this->make($name);
}
}
~~~
# 測試類
~~~
<?php
include './Container.php';
interface TrafficTool
{
public function go();
}
class Train implements TrafficTool
{
public function go()
{
echo "train....";
}
}
class Leg implements TrafficTool
{
public function go()
{
echo "leg..";
}
}
class Traveller
{
/**
* @var Leg|null|Train
* 旅行工具
*/
protected $_trafficTool;
public function __construct(TrafficTool$trafficTool)
{
$this->_trafficTool = $trafficTool;
}
public function visitTibet()
{
$this->_trafficTool->go();
}
}
//實例化IoC容器
$app = new Container();
//綁定某一功能到IoC
$app->bind('TrafficTool', 'Train');
$app->bind('travellerA', 'Traveller');
// 實例化對象
$tra = $app->make('travellerA');
$tra->visitTibet();
~~~
# 源碼解析
在laravel框架中,服務容器是通過 `Illuminate\Container\Container` 類來實現的,其實現原理和上面相同,這里給出容器類的工作圖
服務綁定有時也稱為服務注冊,在全文兩者意義相同,只是對不同上下文環境某種說法更加貼切而已
服務容器工作示意圖


服務容器類中定義了兩個用于管理服務的屬性,分別是`$bindings`和`$instance`,其中`$bindings`用于存儲提供服務的回調函數,而`$instances`用于存儲程序中的共享實例,也可以稱為單例


這幾個函數實現了laravel框架的服務容器的服務綁定功能.主要是由bind()函數實現的
singleton()函數實現的是單例綁定,即程序中如果沒有服務名稱對應的實例對象,則通過服務容器實例化一個后進行記錄,如果在后續的程序還需要同名的服務時則返回先前創建的服務實例對象.該函數相當于bind()函數的一個特例,即參數`$shared`值為true的情況










- 書列表
- laravel框架關鍵技術
- 第一章 組件化開發與composer使用
- 簡介
- composer
- 添加路由組件
- 添加控制器模塊
- 添加模型組件
- 添加視圖組件
- 第三章 laravel框架中常用的php語法
- 匿名函數
- 文件包含
- 魔術方法
- 魔術常量
- 反射
- 后期靜態綁定
- traits
- 第四章 laravel框架中使用的HTTP協議基礎
- HTTP協議
- 數據庫
- 數據遷移
- 第六章 laravel框架中的設計模式
- IOC模式
- php核心技術與最佳實踐
- 第一章面向對象核心
- 反射
- 簡單ORM
- 異常和錯誤
- 接口
- 第二章,面向對象設計
- 設計原則
- 單一職責
- 接口隔離
- 開放封閉
- 替換原則
- 依賴倒置
- linux是怎么寫的呢?
- 第三章 正則表達
- 認識正則
- 第四章 php網絡技術應用
- HTTP協議詳解
- php和http相關函數
- 垃圾信息防御措施
- 現代操作系統
- 引論
- sql必知必會
- 限制結果
- 按位置排序
- where求職順序
- IN操作符
- like
- 函數
- group by
- 組合查詢
- 插入檢索出的數據
- 視圖
- 高性能mysql
- 第一章節 mysql架構與歷史
- mysql架構邏輯圖
- 連接與管理
- 優化與運行
- 讀寫鎖
- 鎖粒度
- 表鎖(table lock)
- 行級鎖(row lock)
- ACID
- 隔離級別
- 死鎖
- 隱式和顯式鎖定
- 多版本并發控制
- Innodb概覽
- 第四章節 Schema與數據類型優化
- 選擇優化的數據類型
- 日期和時間類型
- 標識列
- 特殊類型數據
- 表設計中的缺陷
- 范式
- 計數器表
- 第五章 創建高性能索引
- 索引基礎
- 索引類型
- 索引的優點
- 高性能索引策略
- 選擇合適的索引列順序
- 聚簇索引
- 順序的主鍵什么時候會造成更壞的后果
- 覆蓋索引
- 使用索引掃描來做排序
- 壓縮索引
- 冗余和重復索引
- 索引和鎖
- 支持多種過濾條件
- 什么是范圍條件
- 優化排序
- 維護索引和表
- 表損壞
- 減少索引和數據的碎片
- 第六章 查詢性能優化
- 掃描的行數和訪問類型
- 重構查詢方式
- 查詢執行的基礎
- 重構-改善既有代碼設計
- 第一章-重構
- 什么是重構
- 第一個案列
- 重構第一步
- 王垠博客
- 多態取代價格相關邏輯