摘要:一個(gè)動(dòng)作類(lèi),應(yīng)該有一個(gè)能夠說(shuō)明其功能的名字,比如等。如果需要對(duì)我們的動(dòng)作類(lèi)實(shí)現(xiàn)某種適配器模式,這是非常方便的。使用單動(dòng)作類(lèi)的好處小巧而單一的邏輯域能夠防止代碼重復(fù)并提高代碼的可重用性,保持穩(wěn)定。易于針對(duì)各種場(chǎng)景進(jìn)行獨(dú)立測(cè)試。
當(dāng)我們談?wù)摰綉?yīng)用程序的架構(gòu)的時(shí)候,經(jīng)常會(huì)問(wèn)到一個(gè)經(jīng)典的問(wèn)題,那就是“這段代碼應(yīng)該放在哪里比較好”。 因?yàn)?Laravel 是一個(gè)相當(dāng)靈活的框架,所以要回答這個(gè)問(wèn)題其實(shí)沒(méi)那么容易。我應(yīng)該把我的業(yè)務(wù)邏輯寫(xiě)在 Model 層,還是 Controller 層,或者是其他地方?
當(dāng)你的應(yīng)用程序僅有一個(gè)接入點(diǎn),把業(yè)務(wù)邏輯寫(xiě)在 Controller 層是可以的。但是現(xiàn)在更普遍的的情形是,有很多接入點(diǎn)去調(diào)用相同的功能模塊。
比如說(shuō),太多數(shù)的應(yīng)用程序都有用戶注冊(cè)的功能,它的流程是調(diào)用一個(gè)控制器然后返回一個(gè)注冊(cè)成功或者失敗的視圖。假如這個(gè)應(yīng)用程序還有移動(dòng)端,那就很可能要提供一套針對(duì)移動(dòng)端用戶注冊(cè)的 API ,因?yàn)樗枰祷氐臄?shù)據(jù)格式是 JSON 。而且利用 Laravel 的 artisan 命令來(lái)創(chuàng)建用戶也很常見(jiàn),尤其是在項(xiàng)目前期的開(kāi)發(fā)階段。
上面這兩段代碼可能看起來(lái)沒(méi)有什么問(wèn)題的,但是,隨著業(yè)務(wù)邏輯的增加,就會(huì)顯得代碼很冗余。舉個(gè)例子,如果你需要新用戶注冊(cè)完之后,增加給用戶發(fā)送郵件通知的功能,你必須要再上面兩個(gè)控制器中都添加發(fā)送郵件的代碼。但是如果要保持代碼的簡(jiǎn)潔優(yōu)雅,我們可以把這些業(yè)務(wù)邏輯寫(xiě)到其他地方。
對(duì)于“把業(yè)務(wù)邏輯代碼寫(xiě)到哪里”的這個(gè)問(wèn)題,你去任何論壇都可以得到一個(gè)普遍的答案,那就是 “使用一個(gè) service 層,然后在 controller 層調(diào)用這個(gè)服務(wù)類(lèi)”。是的,沒(méi)錯(cuò),問(wèn)題是我們應(yīng)該怎么設(shè)計(jì) service 類(lèi)?是創(chuàng)建一個(gè) UserService 類(lèi)來(lái)實(shí)現(xiàn)所有跟用戶用戶有關(guān)的業(yè)務(wù)邏輯,然后把這個(gè)類(lèi)注入到需要用到的 Controller 層?或者是還有其他方案?
避免神類(lèi)的坑首先,可以嘗試為一個(gè)特定的模型創(chuàng)建一個(gè)單一類(lèi),其中包含所有的代碼。例如:
看起來(lái)很完美:我們可以任何控制器中申明或者使用 create/delete 方法,并且得到我們想要的結(jié)果。但是,這種實(shí)現(xiàn)有什么問(wèn)題呢? 那就是我們?cè)诮鉀Q問(wèn)題的過(guò)程通常很少使用單一的模型? 。
比如說(shuō),當(dāng)我們給一個(gè)用戶創(chuàng)建了賬號(hào)的時(shí)候,也要同時(shí)給用戶多帶帶創(chuàng)建一個(gè) blog 。如果按照當(dāng)前的方式去實(shí)現(xiàn)這個(gè)流程,我們就必須創(chuàng)建一個(gè) BlogService 類(lèi),然后將其依賴注入到 UserService 類(lèi)。
顯而易見(jiàn),隨著應(yīng)用程序的業(yè)務(wù)的增長(zhǎng),將會(huì)有幾十到上百個(gè) service 類(lèi),其中的一些 service 類(lèi)需要依賴 5 到 6 個(gè)其他 service 類(lèi),最終的結(jié)果就是,出現(xiàn)代碼的冗余跟混亂的局面,而這個(gè)局面是我們想不惜一切代價(jià)去避免的。
介紹單動(dòng)作類(lèi)那么,如果不是用一個(gè)單一的服務(wù)類(lèi)加上幾個(gè)方法,我們決定把它分成幾個(gè)類(lèi)?下面是我最近每一個(gè)項(xiàng)目都采用的方法,結(jié)果很不錯(cuò),推薦給大家。
首先,讓我們拋棄過(guò)于籠統(tǒng)和模糊的服務(wù)術(shù)語(yǔ),來(lái)了解一下我們的新動(dòng)作類(lèi),并定義它們是什么以及它們可以做什么。
一個(gè)動(dòng)作類(lèi),應(yīng)該有一個(gè)能夠說(shuō)明其功能的名字,比如:CreateOrder, ConfirmCheckout, DeleteProduct, AddProductToCart等。
它應(yīng)該有且只有一個(gè)公共方法,作為 API 。理想的情況下,應(yīng)該是相同的方法名,像 handle() 或者 execute() 。如果需要對(duì)我們的動(dòng)作類(lèi)實(shí)現(xiàn)某種適配器模式,這是非常方便的。
它必須對(duì)請(qǐng)求和響應(yīng)不可知。它不處理請(qǐng)求,也不發(fā)送響應(yīng)。這樣的職責(zé)應(yīng)該由控制器來(lái)承擔(dān)。
它可以依賴其它的動(dòng)作類(lèi)。
如果有任何事情阻止它執(zhí)行和/或返回期望的值,那么它必須通過(guò)拋出一個(gè) Exception 來(lái)強(qiáng)制執(zhí)行相關(guān)的業(yè)務(wù)邏輯,并且讓調(diào)用者(或者 Laravel 的 ExceptionHandler )來(lái)承擔(dān)如何呈現(xiàn)/響應(yīng)異常的責(zé)任。
創(chuàng)建我們的 CreateUser 動(dòng)作類(lèi)現(xiàn)在,讓我們看看前面的例子,并用一個(gè)單動(dòng)作類(lèi)來(lái)重構(gòu)它,我們將命名為 CreateUser 。
你或許想知道當(dāng)郵箱地址已經(jīng)被占用時(shí),該方法為什么會(huì)拋出了異常。 這難道不是請(qǐng)求驗(yàn)證來(lái)保證的嗎?當(dāng)然可以。然而,在動(dòng)作類(lèi)內(nèi)部來(lái)執(zhí)行業(yè)務(wù)邏輯不是更好嗎?這樣使得邏輯變得易于理解和調(diào)試。
讓我們看看使用我們動(dòng)作類(lèi)之后的控制器代碼,如下:
現(xiàn)在,無(wú)論我們做什么修改,用戶注冊(cè)過(guò)程都會(huì)由 API 和 Web 版本處理,優(yōu)雅整潔。
動(dòng)作類(lèi)的嵌套假如,我們需要一個(gè)動(dòng)作類(lèi)將 1000 個(gè)用戶導(dǎo)入我們的應(yīng)用中。我們可以寫(xiě)一個(gè)動(dòng)作類(lèi),并且繼續(xù)使用上文的 CreateUser 類(lèi):
非常整潔,不是嗎?我們可以通過(guò)將其嵌入在?Collection::map() 方法中來(lái)重用 CreateUser 代碼,然后返回所有新建用戶的集合。當(dāng)郵件被占用的時(shí)候,我們可以通過(guò)返回 Null Object 或者在 Log 文件中記錄一下,你應(yīng)該已經(jīng)想到了。
動(dòng)作類(lèi)的裝飾現(xiàn)在,假設(shè)我們想在日志中記錄每一個(gè)新注冊(cè)的用戶。我們可以將代碼寫(xiě)在動(dòng)作類(lèi)內(nèi)部,也可以使用裝飾者模式。
然后,我們可以使用 Laravel 的 IoC 容器將?LogCreateUser?類(lèi)綁定到 CreateUser?類(lèi),所有每當(dāng)我們需要一個(gè)后者的實(shí)例時(shí),前者都會(huì)注入進(jìn)來(lái):
AppServiceProvider.php
這使得使用配置或環(huán)境變量來(lái)控制日志記錄功能的激活或停用更為方便:
AppServiceProvider.php
總結(jié)使用這個(gè)方法似乎會(huì)需要很多的類(lèi)。當(dāng)然,用戶注冊(cè)僅僅是一個(gè)簡(jiǎn)單的例子,旨在保證代碼的簡(jiǎn)短清晰。一旦項(xiàng)目的復(fù)雜度開(kāi)始增長(zhǎng),動(dòng)作類(lèi)的真正的價(jià)值就越來(lái)越明顯,因?yàn)槟闱逦闹来a所在及其界定。
使用單動(dòng)作類(lèi)的好處:小巧而單一的邏輯域能夠防止代碼重復(fù)并提高代碼的可重用性,保持穩(wěn)定。
易于針對(duì)各種場(chǎng)景進(jìn)行獨(dú)立測(cè)試。
富有意義的命名在大型項(xiàng)目中更容易閱讀。
易于裝飾。
整個(gè)項(xiàng)目的一致性:防止代碼分布在 Controllers、Models 等。
當(dāng)然,這個(gè)方法是基于我過(guò)去幾年使用 Laravel 的一些經(jīng)驗(yàn)和我在一些項(xiàng)目中的實(shí)踐。這對(duì)我真的很有用,現(xiàn)在我甚至在一些中小型項(xiàng)目中使用。
如果你有不同的方法,我非常期待讀一讀。
更多現(xiàn)代化 PHP 知識(shí),請(qǐng)前往 Laravel / PHP 知識(shí)社區(qū)
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://m.specialneedsforspecialkids.com/yun/28789.html
摘要:在中,提示符可能是。框架使用來(lái)執(zhí)行安裝及管理依賴。為了能訪問(wèn)網(wǎng)頁(yè),要啟動(dòng)程序服務(wù)器。在大多數(shù)類(lèi)系統(tǒng)中,包括,命令行提示符是符號(hào)。這兩個(gè)操作分別對(duì)應(yīng)于的和,即創(chuàng)建和讀取。首個(gè)表單要在模板中編寫(xiě)表單,可以使用表單構(gòu)造器。 【摘要】自從ThinkSNS+不使用ThinkPHP框架而使用Laravel框架之后,很多人都說(shuō)技術(shù)門(mén)檻抬高了,其實(shí)你與TS+的距離僅僅只是學(xué)習(xí)一個(gè)新框架而已,所以,我們...
摘要:請(qǐng)求處理器可以理解為僅包含單個(gè)動(dòng)作的控制器,能夠使請(qǐng)求到響應(yīng)的流程更加清晰明確。一個(gè)好的使用例子是路由的請(qǐng)求處理在和框架中。控制器的動(dòng)作被分為多個(gè)獨(dú)立的請(qǐng)求處理器類(lèi),分別負(fù)責(zé)響應(yīng)單一的動(dòng)作。您是否應(yīng)當(dāng)使用請(qǐng)求處理器替換所有控制器可能不是。 在過(guò)去幾年中, PHP 開(kāi)發(fā)環(huán)境發(fā)生了很大的變化。我們開(kāi)始使用更多更好的設(shè)計(jì)模式,比如 DRY 和 SOLID) 設(shè)計(jì)模式原則。但為什么我們?nèi)匀辉?..
摘要:過(guò)去一年時(shí)間寫(xiě)了多篇文章來(lái)探討了我認(rèn)為的框架最核心部分的設(shè)計(jì)思路代碼實(shí)現(xiàn)。為了大家閱讀方便,我把這些源碼學(xué)習(xí)的文章匯總到這里。數(shù)據(jù)庫(kù)算法和數(shù)據(jù)結(jié)構(gòu)這些都是編程的內(nèi)功,只有內(nèi)功深厚了才能解決遇到的復(fù)雜問(wèn)題。 過(guò)去一年時(shí)間寫(xiě)了20多篇文章來(lái)探討了我認(rèn)為的Larave框架最核心部分的設(shè)計(jì)思路、代碼實(shí)現(xiàn)。通過(guò)更新文章自己在軟件設(shè)計(jì)、文字表達(dá)方面都有所提高,在剛開(kāi)始決定寫(xiě)Laravel源碼分析地...
閱讀 3165·2021-11-22 09:34
閱讀 2803·2021-09-22 15:28
閱讀 831·2021-09-10 10:51
閱讀 1861·2019-08-30 14:22
閱讀 2328·2019-08-30 14:17
閱讀 2742·2019-08-30 11:01
閱讀 2303·2019-08-29 17:19
閱讀 3671·2019-08-29 13:17