Spring AOP高級——源碼實現(2)Spring AOP中通知器(Advisor)與切麵(Aspect)

来源:http://www.cnblogs.com/yulinfeng/archive/2017/11/15/7841167.html
-Advertisement-
Play Games

本文例子完整源碼地址:https://github.com/yu-linfeng/BlogRepositories/tree/master/repositories/Spring%20AOP%E9%AB%98%E7%BA%A7%E2%80%94%E2%80%94%E6%BA%90%E7%A0%81% ...


本文例子完整源碼地址:https://github.com/yu-linfeng/BlogRepositories/tree/master/repositories/Spring%20AOP%E9%AB%98%E7%BA%A7%E2%80%94%E2%80%94%E6%BA%90%E7%A0%81%E5%AE%9E%E7%8E%B0%EF%BC%882%EF%BC%89Spring%20AOP%E4%B8%AD%E9%80%9A%E7%9F%A5%E5%99%A8%EF%BC%88Advisor%EF%BC%89%E4%B8%8E%E5%88%87%E9%9D%A2%EF%BC%88Aspect%EF%BC%89

 

  之所以還未正式進入Spring AOP的源碼,是因為我在閱讀Spring AOP生成代理對象時遇到了一點小麻煩讓我不得不暫時停止,轉而理清有關Spring AOP中的兩個概念性問題。

  前面的博客里都沒有提到過“通知器”這個概念,在《Spring實戰》書中也只是簡單地說明瞭在xml中<aop:advisor>用於定義一個通知器,此後便沒再說明,而是使用<aop:aspect>定義一個切麵。而在《Spring技術內幕》中有關Spring AOP章節中則是介紹了AOP中三個概念:通知、切點、通知器。在這時,我對“通知器”產生了很大的疑惑,查閱了相關資料並沒有滿意的答案,於是決定自己一探究竟。

  首先來討論定義通知器相關的使用方法。 定義一個通知類,其中包含前置通知和後置通知,註意如果是使用<aop:advisor>定義通知器的方式實現AOP則需要通知類實現Advice介面,前置通知方法對應的是MethodBeforeAdvice,後置通知方法對應的是AfterReturningAdvice。

 1 package com.demo;
 2 
 3 import org.springframework.aop.AfterReturningAdvice;
 4 import org.springframework.aop.MethodBeforeAdvice;
 5 import org.springframework.stereotype.Component;
 6 
 7 import java.lang.reflect.Method;
 8 
 9 /**
10  * Created by Kevin on 2017/11/15.
11  */
12 @Component("advisorTest")
13 public class AdvisorTest implements MethodBeforeAdvice, AfterReturningAdvice{
14 
15     /**
16      * 前置通知
17      * @param method
18      * @param args
19      * @param target
20      * @throws Throwable
21      */
22     @Override
23     public void before(Method method, Object[] args, Object target) throws Throwable {
24         System.out.println("前置通知");
25     }
26 
27     /**
28      * 後置通知
29      * @param returnValue
30      * @param method
31      * @param args
32      * @param target
33      * @throws Throwable
34      */
35     @Override
36     public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable {
37         System.out.println("後置通知");
38     }
39 }

  定義一個需要被代理的目標對象。

 1 package com.demo;
 2 
 3 import org.springframework.stereotype.Component;
 4 
 5 /**
 6  * 目標對象,需要被代理的類及方法
 7  * Created by Kevin on 2017/11/15.
 8  */
 9 @Component("testPoint")
10 public class TestPoint {
11 
12     public void test() {
13         System.out.println("方法調用");
14     }
15 }

  我們要達到的目的就是在test方法調用前和調用後分別列印“前置通知”和“後置通知”。

  applicationContext.xml中定義通知器如下:

 1 <?xml version="1.0" encoding="UTF-8"?>
 2 <beans xmlns="http://www.springframework.org/schema/beans"
 3        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
 4        xmlns:context="http://www.springframework.org/schema/context"
 5        xmlns:aop="http://www.springframework.org/schema/aop"
 6        xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd">
 7 
 8     <context:component-scan base-package="com.demo"/>
 9 
10     <aop:config>
11         <aop:pointcut id="test" expression="execution(* com.demo.TestPoint.test())"/>
12         <aop:advisor advice-ref="advisorTest" pointcut-ref="test"/>
13     </aop:config>
14 
15 </beans>

  最後的運行結果符合預期。那麼問題來了,如果我們只想在定義的這個切點 <aop:pointcut id="test" expression="execution(* com.demo.TestPoint.test())"/>里只配置前置通知,這個時候怎麼辦呢?答案是,通過以上方式是不可以的。也就是說如果通過定義Advisor的方式,在有的地方比較局限,狹隘來講通過定義Advisor通知器的方式,只能定義只有一個通知和一個切入點的切麵。當然一個通知不准確,因為上面可以看到只要實現不同的通知介面即可代理,但如果實現了多個通知介面,而只想使用一個時就不可以了。通知器是一個特殊的切麵。

  接著來討論定義切麵相關的使用方法。 如果使用<aop:aspect>定義切麵的方式,通知類是可以不用實現任何通知介面的,這是很大一個便利。同樣要實現上面例子的功能,定義一個通知類,包括前置通知和後置通知。

 1 package com.demo;
 2 
 3 import org.springframework.stereotype.Component;
 4 
 5 /**
 6  * Created by Kevin on 2017/11/15.
 7  */
 8 @Component("aspectTest")
 9 public class AspectTest {
10 
11     /**
12      * 前置通知
13      */
14     public void doBefore() {
15         System.out.println("前置通知");
16     }
17 
18     /**
19      * 後置通知
20      */
21     public void doAfter() {
22         System.out.println("後置通知");
23     }
24 }

  目標對象和上面的例子一致,緊接著是applicationContext.xml中切麵的配置。

 1 <?xml version="1.0" encoding="UTF-8"?>
 2 <beans xmlns="http://www.springframework.org/schema/beans"
 3        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
 4        xmlns:context="http://www.springframework.org/schema/context"
 5        xmlns:aop="http://www.springframework.org/schema/aop"
 6        xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd">
 7 
 8     <context:component-scan base-package="com.demo"/>
 9 
10     <aop:config>
11         <aop:aspect ref="aspectTest">
12             <aop:pointcut id="test" expression="execution(* com.demo.TestPoint.test())"/>
13             <aop:before method="doBefore" pointcut-ref="test"/>
14             <aop:after-returning method="doAfter" pointcut-ref="test"/>
15         </aop:aspect>
16     </aop:config>
17 </beans>

  可以看到我們通過<aop:aspect>定義了一個切麵,如果只需要前置通知,則只定義<aop:before>就可以了,這和<aop:advisor>是很大的不同,由此可知通過<aop:aspect>定義切麵的方式可以在其中靈活地定義通知,而不必像通知器那樣約束。

  實際上可以這麼說,通知器是一個特殊的切麵。而在最開始那兩篇博客中沒有提到是因為那兩個例子中使用的是AspectJ註解,而在AspectJ註解中並沒有與此對應的概念。

  在實際中用到的<aop:advisor>場景最多的莫過於在Spring中配置事務。除此之外,很少用到,也不建議使用。因為最大的一個問題就是定義通知時需要實現通知介面,這違背了一點Spring“非侵入式”編程的初衷。

  這篇博客穿插在源碼的其中是為了更好的理清Spring AOP中各種概念問題,緣由我在開頭已經說過,接下來就正式開始Spring AOP源碼的解讀。

 

 

這是一個能給程式員加buff的公眾號 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、創建一個Ado.net實體模型 二、根據實體模型創建上下文和實體映射 遇到的問題場景是:模型的屬性“代碼生成策略”如果是 使用“舊的 ObjectContext ”方式時,Linq預設返回的ObjectQuery可以直接賦值給combobox ,如果是使用的T4模板,則系統系統創建的上下文對象為... ...
  • 這兩天領導讓我做個噴泉的效果,要把一個個UserControl從一個位置噴出,然後,最後落在最終需要在的位置。 噴泉效果說白了,就是兩個步驟:1、放大,從0放大到需要的倍數;2、縮小,平移,從放大的倍數還原到UserControl的原始大小,並且定位到最終的位置。 雖然,只有兩步,但是,作為寫動畫的 ...
  • 面試題:寫一個固定容量同步容器,擁有put和get方法,以及getCount方法, 能夠支持2個生產者線程以及10個消費者線程的阻塞調用 有兩種方法 1.使用wait和notify/notifyAll來實現 2.使用Lock和Condition來實現 對比兩種方式,Condition的方式可以更加精 ...
  • 不過大部分手機鬧鐘都不支持這種以小時為單位的周期鬧鈴。所以,我以前每次都是都手動調整鬧鐘時間。總感覺有點 Low!於是,我就寫了個簡單的發郵件的 Lua 腳本,放到樹莓派上作為一個shell命令使用;然後在每周一到周五的9點至23點整點各執行一次發郵件的操作。郵件是發到了我的 QQ 郵箱。收到QQ郵... ...
  • 一萬年太久,只爭朝夕 What JDBC 上部 JDBC(Java DataBase Connectivity)Java 資料庫連接,主要提供編寫 Java 資料庫應用程式的 API 支持 java.sql包中定義的常用的基本的 JDBC API: 類 DriverManager-管理一組 JDBC ...
  • 恢復內容開始 Django 創建第一個項目 本章我們將介紹Django 管理工具及如何使用 Django 來創建項目,第一個項目我們以 HelloWorld 來命令項目。 Django 管理工具 安裝 Django 之後,您現在應該已經有了可用的管理工具 django-admin.py。我們可以使用 ...
  • 一.實現多態所具備的條件有3個: 1.繼承關係 2.子類重寫父類的方法 3.父類的引用指向子類的對象 二.實現一波: 1.編寫Animal類,作為一個父類,有一個name方法,用於給子類重寫. 2.編寫Monkey類繼承Animal類,並重寫父類name方法,擁有自己獨有的climb()方法 3.編 ...
  • 對於所有的Web應用,本質上其實就是一個socket服務端,用戶的瀏覽器其實就是一個socket客戶端。 import socket def f1(request): """ 處理用戶請求,並返回相應的內容 :param request: 用戶請求的所有信息 :return: """ f = ope ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...