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