Spring AOP 中@annotation的兩種寫(xiě)法使用
在 Spring AOP 的開(kāi)發(fā)中,我們經(jīng)??吹竭@樣的切面寫(xiě)法:
@Before("@annotation(com.example.annotation.PayLog)")
public void beforeAnnotation() {
System.out.println("通過(guò) @annotation 攔截帶 @PayLog 注解的方法");
}
或者這樣:
@Before("@annotation(payLog)")
public void beforeAnnotation(PayLog payLog) {
System.out.println("檢測(cè)到支付操作:" + payLog.value());
}
看起來(lái)很相似,但又不太一樣 ??
到底有什么區(qū)別?為什么都能生效?
這篇文章幫你徹底搞清楚!
一、什么是 @annotation
在 AOP(Aspect-Oriented Programming,面向切面編程)中,
我們通過(guò)切點(diǎn)表達(dá)式定義要攔截的方法。
其中,@annotation(...) 是一種非常常見(jiàn)的表達(dá)式,用來(lái)匹配:
“所有帶有指定注解的方法”。
舉個(gè)例子
假設(shè)我們定義了一個(gè)自定義注解 @PayLog:
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface PayLog {
String value() default "默認(rèn)支付";
}
二、寫(xiě)法一:只攔截方法,不讀取注解內(nèi)容
@Aspect
@Component
public class PayAspect {
@Before("@annotation(com.example.annotation.PayLog)")
public void beforeAnnotation() {
System.out.println("通過(guò) @annotation 攔截帶 @PayLog 注解的方法");
}
}
這段切面表示:
“在執(zhí)行任何帶有 @PayLog 注解的方法之前,先執(zhí)行我這個(gè)方法。”
示例:
@Service
public class OrderService {
@PayLog
public void pay() {
System.out.println("執(zhí)行支付邏輯...");
}
}
輸出:
通過(guò) @annotation 攔截帶 @PayLog 注解的方法
執(zhí)行支付邏輯...
?? 解釋:
@annotation(com.example.annotation.PayLog)表示攔截所有被該注解標(biāo)記的方法;beforeAnnotation()方法執(zhí)行時(shí),并不會(huì)接收到注解對(duì)象;- 這種寫(xiě)法適合只想攔截,不需要讀取注解里的內(nèi)容。
三、寫(xiě)法二:攔截 + 獲取注解對(duì)象
@Aspect
@Component
public class PayAspect {
@Before("@annotation(payLog)")
public void beforeAnnotation(PayLog payLog) {
System.out.println("檢測(cè)到支付操作:" + payLog.value());
}
}
這段代碼看起來(lái)只多了一個(gè)參數(shù) payLog,
其實(shí)功能更強(qiáng)大——它可以直接獲取目標(biāo)方法上的注解實(shí)例。
示例
@Service
public class OrderService {
@PayLog("微信支付")
public void payWx() {
System.out.println("執(zhí)行微信支付邏輯...");
}
@PayLog("支付寶支付")
public void payAli() {
System.out.println("執(zhí)行支付寶支付邏輯...");
}
}
輸出結(jié)果:
檢測(cè)到支付操作:微信支付
執(zhí)行微信支付邏輯...
檢測(cè)到支付操作:支付寶支付
執(zhí)行支付寶支付邏輯...
?? 解釋:
@annotation(payLog)告訴 AOP:
攔截時(shí)把目標(biāo)方法上的@PayLog實(shí)例賦值給payLog參數(shù);- 于是我們可以直接通過(guò)
payLog.value()獲取注解中的值。
四、兩種寫(xiě)法的對(duì)比總結(jié)
| 對(duì)比項(xiàng) | 寫(xiě)法一 | 寫(xiě)法二 |
|---|---|---|
| 表達(dá)式 | @annotation(com.example.annotation.PayLog) | @annotation(payLog) |
| 方法參數(shù) | 無(wú) | 有(注解類(lèi)型參數(shù)) |
| 是否能讀取注解內(nèi)容 | ? 否 | ? 可以 |
| 主要用途 | 只做攔截、執(zhí)行前后邏輯 | 需要讀取注解參數(shù)(如描述、類(lèi)型等) |
| 示例應(yīng)用 | 簡(jiǎn)單記錄日志 | 根據(jù)注解參數(shù)執(zhí)行不同邏輯 |
五、完整運(yùn)行示例
1、自定義注解
package com.example.annotation;
import java.lang.annotation.*;
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface PayLog {
String value() default "默認(rèn)支付類(lèi)型";
}
2、AOP 切面類(lèi)
package com.example.aspect;
import com.example.annotation.PayLog;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.springframework.stereotype.Component;
@Aspect
@Component
public class PayAspect {
// 寫(xiě)法一:只攔截
@Before("@annotation(com.example.annotation.PayLog)")
public void onlyIntercept() {
System.out.println("攔截到帶 @PayLog 注解的方法");
}
// 寫(xiě)法二:攔截 + 獲取注解
@Before("@annotation(payLog)")
public void beforeWithAnnotation(PayLog payLog) {
System.out.println("檢測(cè)到支付操作:" + payLog.value());
}
}
3、被攔截的業(yè)務(wù)類(lèi)
package com.example.service;
import com.example.annotation.PayLog;
import org.springframework.stereotype.Service;
@Service
public class OrderService {
@PayLog("微信支付")
public void payWx() {
System.out.println("執(zhí)行微信支付邏輯...");
}
@PayLog("支付寶支付")
public void payAli() {
System.out.println("執(zhí)行支付寶支付邏輯...");
}
}
4、輸出結(jié)果
攔截到帶 @PayLog 注解的方法
檢測(cè)到支付操作:微信支付
執(zhí)行微信支付邏輯...攔截到帶 @PayLog 注解的方法
檢測(cè)到支付操作:支付寶支付
執(zhí)行支付寶支付邏輯...
六、總結(jié)
@annotation 有兩種寫(xiě)法:
- @annotation(注解類(lèi)路徑):只攔截,不讀參數(shù);
- @annotation(變量名) + 方法參數(shù):攔截并獲取注解對(duì)象。
兩者都對(duì),只是用途不同。
| 場(chǎng)景 | 推薦寫(xiě)法 |
|---|---|
| 只想攔截注解方法,不關(guān)心內(nèi)容 | @annotation(com.xxx.PayLog) |
| 需要讀取注解參數(shù)(如 type、desc) | @annotation(payLog) + PayLog payLog |
| 想統(tǒng)一記錄操作日志 | 推薦帶參數(shù)寫(xiě)法,靈活性更強(qiáng) |
到此這篇關(guān)于Spring AOP 中@annotation的兩種寫(xiě)法使用的文章就介紹到這了,更多相關(guān)Spring AOP @annotation內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
springboot快速搭建ftpserver服務(wù)端的詳細(xì)步驟
基于springboot,使用ftpserver快速搭建一個(gè)FTP服務(wù)端,搭建過(guò)程很簡(jiǎn)單,我們把過(guò)程分成4個(gè)步驟,一分鐘內(nèi)快速完成構(gòu)建,感興趣的朋友跟隨小編一起看看吧2023-11-11
深入解析Spring?AI框架如何在Java應(yīng)用中實(shí)現(xiàn)智能化交互的關(guān)鍵
本文詳細(xì)介紹了SpringAI框架在Java應(yīng)用中的應(yīng)用,包括實(shí)體類(lèi)映射、函數(shù)回調(diào)等核心功能的實(shí)現(xiàn),通過(guò)源碼分析,幫助開(kāi)發(fā)者更好地理解和使用這些高級(jí)特性,提升業(yè)務(wù)效率,感興趣的朋友跟隨小編一起看看吧2024-11-11
Eclipse?2022?設(shè)置中文漢化的超詳細(xì)圖文教程
這篇文章主要介紹了Eclipse?2022?設(shè)置中文漢化的超詳細(xì)圖文教程,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-03-03
Java常見(jiàn)基礎(chǔ)數(shù)據(jù)結(jié)構(gòu)
這篇文章主要介紹了Java常見(jiàn)數(shù)據(jù)結(jié)構(gòu)面試題,帶有答案及解釋,希望對(duì)廣大的程序愛(ài)好者有所幫助,同時(shí)祝大家有一個(gè)好成績(jī),需要的朋友可以參考下,希望可以幫助到你2021-07-07
Mybatis攔截器如何實(shí)現(xiàn)數(shù)據(jù)權(quán)限過(guò)濾
本文介紹了MyBatis攔截器的使用,通過(guò)實(shí)現(xiàn)Interceptor接口對(duì)SQL進(jìn)行處理,實(shí)現(xiàn)數(shù)據(jù)權(quán)限過(guò)濾功能,通過(guò)在本地線程變量中存儲(chǔ)數(shù)據(jù)權(quán)限相關(guān)信息,并在攔截器的intercept方法中進(jìn)行SQL增強(qiáng)處理2024-12-12
如何基于SpringMVC實(shí)現(xiàn)斷點(diǎn)續(xù)傳(HTTP)
這篇文章主要介紹了如何基于SpringMVC實(shí)現(xiàn)斷點(diǎn)續(xù)傳(HTTP),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-01-01
spring源碼閱讀--@Transactional實(shí)現(xiàn)原理講解
這篇文章主要介紹了spring源碼閱讀--@Transactional實(shí)現(xiàn)原理講解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09
java中Socket設(shè)置超時(shí)時(shí)間的兩種方式
這篇文章主要介紹了java中Socket設(shè)置超時(shí)時(shí)間的兩種方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-11-11

