OOP和AOP都是一種編程思想,在很多主流框架中都有應(yīng)用。OOP是指面向?qū)ο缶幊蹋珹OP是面向切面編程。AOP是對OOP的擴(kuò)展,使用AOP可以更方便的對業(yè)務(wù)代碼進(jìn)行解耦,從而提高代碼質(zhì)量和增加代碼的可重用性。
舉個例子,我們有一個訂單服務(wù),如下:
為了定位問題,我們想在訂單服務(wù)里添加日志記錄功能。在傳統(tǒng)的 OOP 思想下,我們要實現(xiàn)所需的功能,可以創(chuàng)建一個類繼承 OrderService,然后重寫 generateOrder 方法,最后在所有使用 OrderService 的地方替換成新建的類。如下:
最后,在我們調(diào)用 OrderService 的地方,將 OrderService 替換為 InheritOrderService:
至此,經(jīng)過上方的調(diào)整后,滿足了我們的業(yè)務(wù)需求。現(xiàn)在我們回顧一下,過程似乎非常繁瑣,耦合嚴(yán)重,甚至污染了 generateOrder 方法。如果項目中存在 100 處 OrderService 類的調(diào)用,我們就得找到這 100 個地方進(jìn)行修改、替換,這就是 OOP 的思想。
我們知道 OOP 實際上就是對我們的功能屬性、方法做一個抽象封裝,能夠清晰的劃分邏輯單元。但是 OOP 只能夠進(jìn)行縱向的抽象封裝,無法很好的解決 橫向 的重復(fù)代碼,而 AOP 則很好的解決了這一問題。
如上圖所示,我們有兩個類:訂單類 和 用戶類,我們對其相關(guān)功能做了封裝。但是,權(quán)限檢查、日志記錄等功能就是在重復(fù)的編碼,而利用 AOP 思想就可以將這些功能 橫向切 出去,然后在適當(dāng)?shù)臅r候再將這些功能植入進(jìn)來:
這就是 AOP。
AOP 的主要作用就是在不侵入原有代碼的情況下添加新的功能,常常應(yīng)用于中間件、攔截器、裝飾器中。
AOP的實現(xiàn)有幾個關(guān)鍵點(diǎn):
- Joinpoint(連接點(diǎn)),指程序執(zhí)行的某個特定位置,將通知放置的地方。如方法調(diào)用前、方法調(diào)用后、返回、拋出異常等。允許使用通知的地方都可稱為連接點(diǎn)。
- Pointcut(切入點(diǎn)),指需織入目標(biāo)的方法。假設(shè)一個目標(biāo)對象(類)中擁有 10 個方法,需要在其中 3 個方法中植入通知,這 3 個方法稱為切入點(diǎn)。
- Advice(通知),指攔截到 Joinpoint 之后要做的事情,即對切入點(diǎn)增強(qiáng)的內(nèi)容。
- Target(目標(biāo)),指代理的目標(biāo)對象。
- Weaving(植入),指把增強(qiáng)代碼應(yīng)用到目標(biāo)上,生成代理對象的過程。
- Proxy(代理),指生成的代理對象。
- Aspect(切面),切入點(diǎn)和通知的結(jié)合。
總結(jié)一句話就是:選擇合適的標(biāo)簽位,在標(biāo)簽位綁定增強(qiáng)腳本,執(zhí)行腳本中的指定邏輯。