Spring學習之路3-AOP及實現AOP的三種方式

来源:https://www.cnblogs.com/wangjr1994/archive/2020/03/20/12528806.html
-Advertisement-
Play Games

AOP 面向切麵的程式設計(Aspect oriented programming,AOP,又譯作面向方面的程式設計、剖面導向程式設計)是電腦科學中的一種程式設計思想,旨在將橫切關註點與業務主體進行進一步分離,以提高程式代碼的模塊化程度。通過在現有代碼基礎上增加額外的通知(Advice)機制,能夠 ...


AOP

面向切麵的程式設計(Aspect-oriented programming,AOP,又譯作面向方面的程式設計、剖面導向程式設計)是電腦科學中的一種程式設計思想,旨在將橫切關註點與業務主體進行進一步分離,以提高程式代碼的模塊化程度。通過在現有代碼基礎上增加額外的通知(Advice)機制,能夠對被聲明為“切點(Pointcut)”的代碼塊進行統一管理與裝飾,如“對所有方法名以‘set*’開頭的方法添加後臺日誌”。該思想使得開發人員能夠將與代碼核心業務邏輯關係不那麼密切的功能(如日誌功能)添加至程式中,同時又不降低業務代碼的可讀性。面向切麵的程式設計思想也是面向切麵軟體開發的基礎。

Maven 依賴

<!-- https://mvnrepository.com/artifact/org.aspectj/aspectjweaver -->
<dependency>
    <groupId>org.aspectj</groupId>
    <artifactId>aspectjweaver</artifactId>
    <version>1.9.5</version>
</dependency>

實現方式一:原生 Spring API 介面

public interface UserService {
    void add();
    int delete();
    void query();
    int update();
}

public class UserServiceImpl implements UserService {
    @Override
    public void add() {
        System.out.println("function:add()");
    }

    @Override
    public int delete() {
        System.out.println("function:delete()");
        return 0;
    }

    @Override
    public void query() {
        System.out.println("function:query()");
    }

    @Override
    public int update() {
        System.out.println("function:update()");
        return 0;
    }
}

//方法執行前的log
public class BeforeLog implements MethodBeforeAdvice {

    @Override
    public void before(Method method, Object[] args, Object target) throws Throwable {

        System.err.println("method before log" + target.getClass().getName() + method.getName());
    }
}

//方法執行後的log
public class AfterLog implements AfterReturningAdvice {
    @Override
    public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable {
        System.err.println("method before log" + target.getClass().getName() + "-->" + method.getName() + "---return:" + returnValue);
    }
}
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd
       http://www.springframework.org/schema/aop
       http://www.springframework.org/schema/aop/spring-aop.xsd">


    <bean id="log1" class="com.youzi.log.BeforeLog"/>
    <bean id="log2" class="com.youzi.log.AfterLog"/>
    <bean id="userservice" class="com.youzi.service.UserServiceImpl"/>

    <!--配置AOP-->
    <aop:config>
        <aop:pointcut id="pointcut1" expression="execution(int com.youzi.service.UserServiceImpl.*(..))"/>
        <aop:pointcut id="pointcut2" expression="execution(void com.youzi.service.*.add(..))"/>
        <aop:advisor advice-ref="log1" pointcut-ref="pointcut1"/>
        <aop:advisor advice-ref="log2" pointcut-ref="pointcut2"/>
    </aop:config>
</beans>

execution() 切點函數語法:

execution(返回類型 包名.類名.方法名(參數類型))
  • 也可以用*號表示所有的類型。

  • 類名也可以用*代替表示包下麵的所有類

  • *(..):這個星號表示方法名,*號表示所有的方法,括弧中是方法的參數,兩個點表示任何參數。

實現方式二:自定義類

第一種實現方式雖然實現了在方法前或者方法後添加日誌,但是每實現一個功能就需要實現一個介面,通過下麵的方法可以把這些方法寫在一個類中

public class MethodLog {
    public void beforeMethod() {
        System.out.println("方法執行前");
    }

    public void afterMethod() {
        System.out.println("方法執行後");
    }
}
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
                           http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd">

    <bean id="MethodLog" class="com.youzi.log.MethodLog"/>
    <bean id="userservice" class="com.youzi.service.UserServiceImpl"/>

    <aop:config>
        <aop:aspect ref="MethodLog">
            <aop:pointcut id="poindcut1" expression="execution(* com.youzi.service.*.*(..))"/>
            <aop:before method="beforeMethod" pointcut-ref="poindcut1"/>
            <aop:after method="afterMethod" pointcut-ref="poindcut1"/>
        </aop:aspect>
    </aop:config>
</beans>

上面的方法雖然簡化了代碼,但是目前看來不能像第一種方式一樣通過反射取到執行方法的信息

實現方式三:使用註解

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
                           http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd">

    <bean id="MethodLog" class="com.youzi.log.MethodLog"/>
    <bean id="userservice" class="com.youzi.service.UserServiceImpl"/>
    <!--在xml中聲明使用註解的方式-->
    <aop:aspectj-autoproxy/>
</beans>
@Aspect 
public class MethodLog {
    @Before("execution(* com.youzi.service.UserServiceImpl.*(..))")
    public void beforeMethod() {
        System.out.println("方法執行前1");
    }

    @After("execution(* com.youzi.service.UserServiceImpl.*(..))")
    public void afterMethod() {
        System.out.println("方法執行後1");
    }

    @Around("execution(* com.youzi.service.UserServiceImpl.update(..))")
    public int aroundMethod(ProceedingJoinPoint joinPoint) throws Throwable {
        System.err.println("環繞前");
        System.err.println(joinPoint.getSignature());
        Object proceed = joinPoint.proceed();
        System.err.println("環繞後");
        return Integer.valueOf((Integer) proceed);
    }

}

測試代碼統一為

ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService = context.getBean("userservice", UserService.class);
userService.query();
userService.add();
userService.delete();
userService.update();

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

-Advertisement-
Play Games
更多相關文章
  • 一、程式結構分析 第一次作業 1.設計構思與自我評價 第一次作業難度不是太高,只需要解析由常數、冪函數簡單組合的表達式。其中每一項都可以看做coef*x**index的固定形式,建立包含coef,index兩個Biginteger屬性的Term類作為最基本的數據結構,並且實現單項求導的方法。在Ter ...
  • 1、Spring是什麼? Spring是一個輕量級的IoC和AOP容器框架。是為Java應用程式提供基礎性服務的一套框架,目的是用於簡化企業應用程式的開發,它使得開發者只需要關心業務需求。常見的配置方式有三種:基於XML的配置、基於註解的配置、基於Java的配置。 主要由以下幾個模塊組成: Spri ...
  • SpringMVC的工作原理圖: SpringMVC流程 1、 用戶發送請求至前端控制器DispatcherServlet。 2、 DispatcherServlet收到請求調用HandlerMapping處理器映射器。 3、 處理器映射器找到具體的處理器(可以根據xml配置、註解進行查找),生成處 ...
  • 成為詩人後可能不需要刻意地按照某種模式去創作,但成為詩人前他們一定是認真地研究過成百上千的唐詩宋詞、古今名句。 ...
  • 心之所向,勇往直前!記錄開發過程中的那些小事,給自己加點經驗值。 前言 上篇《.netCore + Vue框架搭建之旅》前端篇:微前端架構設計》講了一下設計的核心思想,這篇主要說一下遠程載入模塊(動態載入js、css)。整個架構主要利用這個方式來實現,所以就單獨再抽出來記錄。 上篇講的是通過jque ...
  • 原文地址: http://www.yiidian.com/jdbc/jdbc jdbcrowset.html 1 什麼是JDBC的RowSet? 與ResultSet相比,RowSet預設是可滾動、可更新、可序列化的結果集,而且作為JavaBean使用,因此能方便的在網路上傳輸,用於同步兩端的數據, ...
  • 前言 本博客是長篇系列博客,旨在幫助想提升自己,突破技術瓶頸,但又苦於不知道如何進行系統學習從而提升自己的童鞋.筆者假設讀者具有3 5年開發經驗,java基礎扎實,想突破自己的技術瓶頸,成為一位優秀的架構師,所謂java基礎扎實,比如: 1.java語言三大特性. 2.java語言八大基本類型及其表 ...
  • finally代碼塊:定義一定執行的代碼。 通常用於關閉資源連接。 三種格式: try { }catch (){ }//*****try { }finally { }//*****可以多個catchtry { }catch (){ }finally { } catch:處理異常,沒有catch就代表 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...