細說Spring——AOP詳解(AOP概覽)

1、對AOP的初印象

首先先給出一段比較專業的術語(來自百度):web

在軟件業,AOP爲Aspect Oriented Programming的縮寫,意爲:面向切面編程,經過預編譯方
式和運行期動態代理實現程序功能的統一維護的一種技術。AOP是OOP的延續,是軟件開發中的一個
熱點,也是Spring框架中的一個重要內容,是函數式編程的一種衍生範型。利用AOP能夠對業務邏輯
的各個部分進行隔離,從而使得業務邏輯各部分之間的耦合度下降,提升程序的可重用性,同時提升
了開發的效率。

而後咱們舉一個比較容易理解的例子(來自:Spring 之 AOP):正則表達式

要理解切面編程,就須要先理解什麼是切面。用刀把一個西瓜分紅兩瓣,切開的切口就是切面;炒菜,鍋與爐子共同來完成炒菜,鍋與爐子就是切面。web層級設計中,web層->網關層->服務層->數據層,每一層之間也是一個切面。編程中,對象與對象之間,方法與方法之間,模塊與模塊之間都是一個個切面。編程

咱們通常作活動的時候,通常對每個接口都會作活動的有效性校驗(是否開始、是否結束等等)、以及這個接口是否是須要用戶登陸。框架

按照正常的邏輯,咱們能夠這麼作。
這裏寫圖片描述svg

這有個問題就是,有多少接口,就要多少次代碼copy。對於一個「懶人」,這是不可容忍的。好,提出一個公共方法,每一個接口都來調用這個接口。這裏有點切面的味道了。
這裏寫圖片描述函數式編程

一樣有個問題,我雖然不用每次都copy代碼了,可是,每一個接口總得要調用這個方法吧。因而就有了切面的概念,我將方法注入到接口調用的某個地方(切點)。函數

這裏寫圖片描述

這樣接口只須要關心具體的業務,而不須要關注其餘非該接口關注的邏輯或處理。
紅框處,就是面向切面編程。設計

2、AOP中的相關概念

看過了上面的例子,我想你們腦中對AOP已經有了一個大體的雛形,可是又對上面提到的切面之類的術語有一些模糊的地方,接下來就來說解一下AOP中的相關概念,瞭解了AOP中的概念,才能真正的掌握AOP的精髓。
這裏仍是先給出一個比較專業的概念定義代理

  • Aspect(切面): Aspect 聲明相似於 Java 中的類聲明,在 Aspect 中會包含着一些 Pointcut 以及相應的 Advice。
  • Joint point(鏈接點):表示在程序中明肯定義的點,典型的包括方法調用,對類成員的訪問以及異常處理程序塊的執行等等,它自身還能夠嵌套其它 joint point。
  • Pointcut(切點):表示一組 joint point,這些 joint point 或是經過邏輯關係組合起來,或是經過通配、正則表達式等方式集中起來,它定義了相應的 Advice 將要發生的地方。
  • Advice(加強):Advice 定義了在 Pointcut 裏面定義的程序點具體要作的操做,它經過 before、after 和 around 來區別是在每一個 joint point 以前、以後仍是代替執行的代碼。
  • Target(目標對象):織入 Advice 的目標對象.。
  • Weaving(織入):將 Aspect 和其餘對象鏈接起來, 並建立 Adviced object 的過程

而後舉一個容易理解的例子
看完了上面的理論部分知識, 我相信仍是會有很多朋友感受到 AOP 的概念仍是很模糊, 對 AOP 中的各類概念理解的還不是很透徹. 其實這很正常, 由於 AOP 中的概念是在是太多了, 我當時也是花了老大勁才梳理清楚的.
下面我以一個簡單的例子來比喻一下 AOP 中 Aspect, Joint point, PointcutAdvice之間的關係.
讓咱們來假設一下, 從前有一個叫爪哇的小縣城, 在一個月黑風高的晚上, 這個縣城中發生了命案. 做案的兇手十分狡猾, 現場沒有留下什麼有價值的線索. 不過萬幸的是, 剛從隔壁回來的老王剛好在這時候無心中發現了兇手行兇的過程, 可是因爲天色已晚, 加上兇手蒙着面, 老王並無看清兇手的面目, 只知道兇手是個男性, 身高約七尺五寸. 爪哇縣的縣令根據老王的描述, 對守門的士兵下命令說: 凡是發現有身高七尺五寸的男性, 都要抓過來審問. 士兵固然不敢違背縣令的命令, 只好把進出城的全部符合條件的人都抓了起來.code

來讓咱們看一下上面的一個小故事和 AOP 到底有什麼對應關係.
首先咱們知道, 在 Spring AOP 中 Joint point 指代的是全部方法的執行點, 而 point cut 是一個描述信息, 它修飾的是 Joint point, 經過 point cut, 咱們就能夠肯定哪些 Joint point 能夠被織入 Advice. 對應到咱們在上面舉的例子, 咱們能夠作一個簡單的類比, Joint point 就至關於 爪哇的小縣城裏的百姓,pointcut 就至關於 老王所作的指控, 即兇手是個男性, 身高約七尺五寸, Advice 則是施加在符合老王所描述的嫌疑人的動做: 抓過來審問.
爲何能夠這樣類比呢?

  • Joint point : 爪哇的小縣城裏的百姓: 由於根據定義, Joint point 是全部可能被織入 Advice 的候選的點, 在 Spring AOP中, 則能夠認爲全部方法執行點都是 Joint point. 而在咱們上面的例子中, 命案發生在小縣城中, 按理說在此縣城中的全部人都有多是嫌疑人.

  • Pointcut :男性, 身高約七尺五寸: 咱們知道, 全部的方法(joint point) 均可以織入 Advice, 可是咱們並不但願在全部方法上都織入 Advice, 而 Pointcut 的做用就是提供一組規則來匹配joinpoint, 給知足規則的 joinpoint 添加 Advice. 同理, 對於縣令來講, 他再昏庸, 也知道不能把縣城中的全部百姓都抓起來審問, 而是根據兇手是個男性, 身高約七尺五寸, 把符合條件的人抓起來. 在這裏 兇手是個男性, 身高約七尺五寸 就是一個修飾謂語, 它限定了兇手的範圍, 知足此修飾規則的百姓都是嫌疑人, 都須要抓起來審問.

  • Advice :抓過來審問, Advice 是一個動做, 即一段 Java 代碼, 這段 Java 代碼是做用於 point cut 所限定的那些 Joint point 上的. 同理, 對比到咱們的例子中, 抓過來審問 這個動做就是對做用於那些知足 男性, 身高約七尺五寸 的爪哇的小縣城裏的百姓.

  • Aspect::Aspect 是 point cut 與 Advice 的組合, 所以在這裏咱們就能夠類比: 「根據老王的線索, 凡是發現有身高七尺五寸的男性, 都要抓過來審問」 這一整個動做能夠被認爲是一個 Aspect.

最後是一個描述這些概念之間關係的圖
這裏寫圖片描述

3、其餘的一些內容

AOP中的Joinpoint能夠有多種類型:構造方法調用,字段的設置和獲取,方法的調用,方法的執行,異常的處理執行,類的初始化。也就是說在AOP的概念中咱們能夠在上面的這些Joinpoint上織入咱們自定義的Advice,可是在Spring中卻沒有實現上面全部的joinpoint,確切的說,Spring只支持方法執行類型的Joinpoint

Advice 的類型

  • before advice, 在 join point 前被執行的 advice. 雖然 before advice 是在 join point 前被執行, 可是它並不可以阻止 join point 的執行, 除非發生了異常(即咱們在 before advice 代碼中, 不能人爲地決定是否繼續執行 join point 中的代碼)

  • after return advice, 在一個 join point 正常返回後執行的 advice

  • after throwing advice, 當一個 join point 拋出異常後執行的 advice
  • after(final) advice, 不管一個 join point 是正常退出仍是發生了異常, 都會被執行的 advice.
  • around advice, 在 join point 前和 joint point 退出後都執行的 advice. 這個是最經常使用的 advice.
  • introduction,introduction能夠爲原有的對象增長新的屬性和方法。

Spring中,經過動態代理和動態字節碼技術實現了AOP,這些內容,咱們將在之後進行講解。