摘要:依賴注入依賴注入一詞是由提出的術語,它是將組件注入到應用程序中的一種行為。就像說的依賴注入是敏捷架構中關鍵元素。類依賴于,所以我們的代碼可能是這樣的創建一個這是一種經典的方法,讓我們從使用構造函數注入開始。
文章轉自:https://learnku.com/laravel/t...
作為開發者,我們一直在嘗試通過使用設計模式和嘗試新的健壯型框架來尋找新的方式來編寫設計良好且健壯的代碼。在本篇文章中,我們將通過 Laravel 的 IoC 組件探索依賴注入設計模式,并了解它如何改進我們的設計。
依賴注入依賴注入一詞是由?Martin Fowler 提出的術語,它是將組件注入到應用程序中的一種行為。就像?Ward Cunningham?說的:
依賴注入是敏捷架構中關鍵元素。
讓我們看一個例子:
class UserProvider{ protected $connection; public function __construct(){ $this->connection = new Connection; } public function retrieveByCredentials( array $credentials ){ $user = $this->connection ->where( "email", $credentials["email"]) ->where( "password", $credentials["password"]) ->first(); return $user; } }
如果你要測試或者維護這個類,你必須訪問數據庫的實例來進行一些查詢。為了避免必須這樣做,你可以將此類與其他類進行 解耦 ,你有三個選項之一,可以將 Connection 類注入而不需要直接使用它。
將組件注入類時,可以使用以下三個選項之一:
構造方法注入class UserProvider{ protected $connection; public function __construct( Connection $con ){ $this->connection = $con; } ...Setter 方法注入
同樣,我們也可以使用 Setter 方法注入依賴關系:
class UserProvider{ protected $connection; public function __construct(){ ... } public function setConnection( Connection $con ){ $this->connection = $con; } ...接口注入
interface ConnectionInjector{ public function injectConnection( Connection $con ); } class UserProvider implements ConnectionInjector{ protected $connection; public function __construct(){ ... } public function injectConnection( Connection $con ){ $this->connection = $con; } }
當一個類實現了我們的接口時,我們定義了 injectConnection 方法來解決依賴關系。
優勢現在,當測試我們的類時,我們可以模擬依賴類并將其作為參數傳遞。每個類必須專注于一個特定的任務,而不應該關心解決它們的依賴性。這樣,你將擁有一個更專注和可維護的應用程序。
如果你想了解更多關于 DI 的信息,Alejandro Gervassio 在 本系列 文章中對其進行了廣泛而專業的介紹,所以一定要去讀這些文章。那么,什么又是 IoC 呢?IoC (控制反轉)不需要使用依賴注入,但它可以幫助你有效的管理依賴關系。
控制反轉Ioc 是一個簡單的組件,可以更加方便地解析依賴項。你可以將對象形容為容器,并且每次解析類時,都會自動注入依賴項。
Laravel Ioc當你請求一個對象時, Laravel Ioc 在解決依賴關系的方式上有些特殊:
我們使用一個簡單的例子,將在本文中改進它。
SimpleAuth 類依賴于 FileSessionStorage ,所以我們的代碼可能是這樣的:
class FileSessionStorage{ public function __construct(){ session_start(); } public function get( $key ){ return $_SESSION[$key]; } public function set( $key, $value ){ $_SESSION[$key] = $value; } } class SimpleAuth{ protected $session; public function __construct(){ $this->session = new FileSessionStorage; } } //創建一個 SimpleAuth $auth = new SimpleAuth();
這是一種經典的方法,讓我們從使用構造函數注入開始。
class SimpleAuth{ protected $session; public function __construct( FileSessionStorage $session ){ $this->session = $session; } }
現在我們創建一個對象:
$auth = new SimpleAuth( new FileSessionStorage() );
現在我想使用 Laravel Ioc 來管理這一切。
因為?Application?類繼承自?Container?類,所以你可以通過 App 門面來訪問容器。
App::bind( "FileSessionStorage", function(){ return new FileSessionStorage; });
bind 方法第一個參數是要綁定到容器的唯一 ID ,第二個參數是一個回調函數每當執行 FileSessionStorage 類時執行,我們還可以傳遞一個表示類名的字符串,如下所示。
Note: 如果你查看 Laravel 包時,你將看到綁定有時會分組,比如(?view,?view.finder……)。
假設我們將會話存儲轉換為 Mysql 存儲,我們的類應該類似于:
class MysqlSessionStorage{ public function __construct(){ //... } public function get($key){ // do something } public function set( $key, $value ){ // do something } }
現在我們已經更改了依賴項,我們還需要更改 SimpleAuth 構造函數,并將新對象綁定到容器中!
高級模塊不應該依賴于低級模塊,兩者都應該依賴于抽象對象。
抽象不應該依賴于細節,細節應該取決于抽象。Robert C. Martin
我們的 SimpleAuth 類不應該關心我們的存儲是如何完成的,相反它更應該關注于消費的服務。
因此,我們可以抽象實現我們的存儲:
interface SessionStorage{ public function get( $key ); public function set( $key, $value ); }
這樣我們就可以實現并請求 SessionStorage 接口的實例:
class FileSessionStorage implements SessionStorage{ public function __construct(){ //... } public function get( $key ){ //... } public function set( $key, $value ){ //... } } class MysqlSessionStorage implements SessionStorage{ public function __construct(){ //... } public function get( $key ){ //... } public function set( $key, $value ){ //... } } class SimpleAuth{ protected $session; public function __construct( SessionStorage $session ){ $this->session = $session; } }
如果我們使用 App::make("SimpleAuth") 通過容器解析 SimpleAuth
類,容器將會拋出 BindingResolutionException ,嘗試從綁定解析類之后,返回到反射方法并解析所有依賴項。
Uncaught exception "IlluminateContainerBindingResolutionException" with message "Target [SessionStorage] is not instantiable."
容器正試圖將接口實例化。我們可以為該接口做一個具體的綁定。
App:bind( "SessionStorage", "MysqlSessionStorage" );
現在每次我們嘗試從容器解析該接口時,我們會得到一個?MysqlSessionStorage?實例。如果我們想要切換我們的存儲服務,我們只要變更一下這個綁定。
Note:?如果你想要查看一個類是否已經在容器中被綁定,你可以使用?App::bound("ClassName")?,或者可以使用?App::bindIf("ClassName")?來注冊一個還未被注冊過的綁定。
Laravel Ioc 也提供?App::singleton("ClassName", "resolver")?來處理單例的綁定。
你也可以使用?App::instance("ClassName", "instance")?來創建單例的綁定。
如果容器不能解析依賴項就會拋出?ReflectionException ,但是我們可以使用?App::resolvingAny(Closure) 方法以回調函數的形式來解析任何指定的類型。
Note:?如果你為某個類型已經注冊了一個解析方式?resolvingAny?方法仍然會被調用,但它會直接返回?bind?方法的返回值。
小貼士這些綁定寫在哪兒:
如果只是一個小型應用你可以寫在一個全局的起始文件?global/start.php 中,但如果項目變得越來越龐大就有必要使用?Service Provider 。
測試:
當需要快速簡易的測試可以考慮使用?php artisan tinker ,它十分強大,且能幫你提升你的 Laravel 測試流程。
Reflection API:
PHP 的 Reflection API 是非常強大的,如果你想要深入 Laravel Ioc 你需要熟悉 Reflection API ,可以先看下這個?教程?來獲得更多的信息。
和往常一樣,學習或者了解某些東西最好的方法就是查看源代碼。Laravel Ioc 僅僅只是一個文件,不會花費你太多時間來完成所有功能。你想了解更多關于 Laravel Ioc 或者 Ioc 的一般情況嗎?那請告訴我們吧!
文章轉自:https://learnku.com/laravel/t...
更多文章:https://learnku.com/laravel/c...
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/31200.html
摘要:可以為服務提供者的方法設置類型提示。方法將在所有其他服務提供者均已注冊之后調用。所有服務提供者都在配置文件中注冊。可以選擇推遲服務提供者的注冊,直到真正需要注冊綁定時,這樣可以提供應用程序的性能。 本文最早發布于 Rootrl的Blog 導言 Laravel是一款先進的現代化框架,里面有一些概念非常重要。在上手Laravel之前,我認為先弄懂這些概念是很有必要的。你甚至需要重溫下PHP...
摘要:簡述的生命周期采用了單一入口模式,應用的所有請求入口都是文件。分發請求一旦應用完成引導和所有服務提供者都注冊完成,將會移交給路由進行分發。此外,由于對動態方法的獨特用法,也使測試起來非常容易。 本書的 GitHub 地址:https://github.com/todayqq/PH... Laravel 作為現在最流行的 PHP 框架,其中的知識較多,所以單獨拿出來寫一篇。 簡述 La...
摘要:劃下重點,服務容器是用于管理類的依賴和執行依賴注入的工具。類的實例化及其依賴的注入,完全由服務容器自動的去完成。 本文首發于 深入剖析 Laravel 服務容器,轉載請注明出處。喜歡的朋友不要吝嗇你們的贊同,謝謝。 之前在 深度挖掘 Laravel 生命周期 一文中,我們有去探究 Laravel 究竟是如何接收 HTTP 請求,又是如何生成響應并最終呈現給用戶的工作原理。 本章將帶領大...
摘要:如何實現持久化持久化,將在內存中的的狀態保存到硬盤中,相當于備份數據庫狀態。相當于備份數據庫接收到的命令,所有被寫入的命令都是以的協議格式來保存的。 最近社區里面有一篇文章引起了最多程序猿的關注,Laravel、PHPer 面試可能會遇到的問題,看評論區不少小伙伴們被難倒,對于一些問題同樣難倒了我(其實有很多啦),趁著周末有空,又總結梳理了一遍,順便來答一波題。由于個人技術水平有限,答...
閱讀 1314·2019-08-30 15:44
閱讀 1983·2019-08-30 13:49
閱讀 1659·2019-08-26 13:54
閱讀 3492·2019-08-26 10:20
閱讀 3257·2019-08-23 17:18
閱讀 3299·2019-08-23 17:05
閱讀 2135·2019-08-23 15:38
閱讀 1018·2019-08-23 14:35