Spring基于常用AspectJ切點表達(dá)式使用介紹
execution (常用,方法級別的匹配)
語法:
execution(modifiers-pattern? ret-type-pattern declaring-type-pattern?name-pattern(param-pattern) throws-pattern?)
- modifiers-pattern:方法的訪問符,如public、protected、default,不能匹配private方法
- ret-type-pattern:方法的返回值類型,例:java.util.List、java.lang.String(類的全限定名,或者支持的簡寫如String、Integer)
- declaring-type-pattern:方法所在類的全路徑名,例:life.cqq.controller.UserController
- name-pattern:方法名
- param-pattern:方法的參數(shù)類型,例:java.util.List、java.lang.String
- throws-pattern:方法拋出的異常類型,例:java.lang.RuntimeException、java.lang.Exception
(注:返回值類型 & 參數(shù)類型支持泛型判斷)
帶有?的位,非必寫,即可省略。所以,一個表達(dá)式至少由3部分組成。例:
// 匹配所有方法
execution(* *(..))
若該位省略,例modifiers-pattern,意味著匹配該位能匹配的全部方法:public、protected、default修飾的方法
上例子:
@Pointcut("execution(public java.util.List<java.lang.Integer> *.test(java.util.List<java.lang.String>) throws java.lang.RuntimeException)")
匹配所有的方法名為test,方法格式為下方格式的方法。
public List<Iteger> test(List<String> list) throws RuntimeException
通配符
- “..” : 應(yīng)用在declaring-type-pattern上時,意味著掃描子孫包。應(yīng)用在param-pattern時意味著任意類型
- “*” :任意匹配符
- “+” :表示按照類型匹配,必須追加在declaring-type-pattern中的類名后,表示所有的該類型的類、繼承和擴(kuò)展該類型的類
上例子:
// 1. 關(guān)于 .. 通配符
// 匹配子孫包下的所有類的所有方法
execution(* life.cqq..*.*(..))// 2. 關(guān)于 * 通配符
// 匹配所有類的所有方法
execution(* *(..))// 匹配子孫包下帶有指定前綴的包下的所有類的所有方法
execution(* life.cqq..prefix*.*.*(..))// 3. 關(guān)于 + 通配符
// 匹配子孫包下繼承或?qū)崿F(xiàn)了指定類型的類 以及 該類型的本類(若該類型不為接口) 下的所有方法
execution(* life.cqq..type+.*(..))
within (常用,類級別的匹配) 語法:
within(declaring-type-pattern)
execution語法中的declaring-type-pattern部分,從通配符的例子中提取出可應(yīng)用于within的內(nèi)容:
// 1. 關(guān)于 .. 通配符
// 匹配子孫包下的所有類的所有方法
within(life.cqq..*)// 2. 關(guān)于 * 通配符
// 匹配所有類的所有方法
within(*)// 匹配子孫包下帶有指定前綴的包下的所有類的所有方法
within(life.cqq..prefix*.*)// 3. 關(guān)于 + 通配符
// 匹配子孫包下繼承或?qū)崿F(xiàn)了指定類型的類 以及 該類型的本類(若該類型不為接口) 下的所有方法
within(life.cqq..type+)
@annotation (常用,方法級別的匹配)
語法:
@annotation(annotation-type-pattern)
與execution一樣針對于方法,匹配添加了指定注解的方法。
@within (常用,類級別的匹配)
語法:
@within(annotation-type-pattern)
與within一樣針對于類,匹配添加了指定注解的類。
邏輯運算符
- &&
- ||
- !
應(yīng)用在多個Point上組成稍微復(fù)雜的匹配表達(dá)式
上例子:
@Component class AopBean { public void test(String str) { System.out.println("String"); } public void test(Integer integer) { System.out.println("Integer"); } } @Pointcut("within(life.cqq.aop.logicsymbol.AopBean)") public void point1() {} @Pointcut("execution(public void test(java.lang.String))") public void point2() {} @Pointcut("execution(public void test(java.lang.Integer))") public void point3() {} // @Around("point1() && point2()") : 匹配參數(shù)為String類型的方法 // @Around("point1() && (point2() || point3())") : 匹配參數(shù)為Integer、String類型的方法 // @Around("point1() && !point2())") : 匹配參數(shù)為Integer類型的方法
以上僅為平時常用的內(nèi)容,還有其他許多寫法,如:args、@args、target、@target等
一次實際應(yīng)用
需求:
- 基于AOP + 自定義注解的方式實現(xiàn)三個系統(tǒng)的接口請求參數(shù)的記錄。
- 需實現(xiàn)就近原則:優(yōu)先判斷方法上日志注解,若無在判斷方法上的注解。類上添加日志注解時,意味著類的全部方法都需要打印請求參數(shù)。
@Pointcut("execution(* life.cqq..controller.*.*(..))") private void log() {} @Pointcut("@within(life.cqq.common.newlog.annotation.AppOpnLog) || @annotation(life.cqq.common.newlog.annotation.AppOpnLog)") public void appOpnLog() { } @Pointcut("@within(life.cqq.common.newlog.annotation.EsOpnLog) || @annotation(life.cqq.common.newlog.annotation.EsOpnLog)") public void esOpnLog() { } @Pointcut("@within(life.cqq.common.newlog.annotation.WebOpnLog) || @annotation(life.cqq.common.newlog.annotation.WebOpnLog)") public void webOpnLog() { } @Around("appOpnLog() || esOpnLog() || webOpnLog()") public Object opnLogAround(ProceedingJoinPoint point) throws Throwable { // ...... return point.proceed(); }
- 因為是三個系統(tǒng),log切點表達(dá)式中的controller前需要使用通配符"…"。因為每個系統(tǒng)的controller包名都是不一致的,但可以確定都符合格式: life.cqq.xxx.controller。
- 基于注解的形式,且實現(xiàn)就近原則,那么既要匹配類上的日志注解也要匹配方法上的日志注解,故使用@within 與 @annotation 并用邏輯或連接。具體是在方法還是在類上添加的注解,在Around方法中解析處理。
到此這篇關(guān)于Spring基于常用AspectJ切點表達(dá)式使用介紹的文章就介紹到這了,更多相關(guān)Spring AspectJ內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot內(nèi)置數(shù)據(jù)源的持久化與解決方案
數(shù)據(jù)源的配置 我們先基于SpringBoot默認(rèn)的HikariDataSource數(shù)據(jù)源,導(dǎo)入JDBC場景,看看SpringBoot幫我們自動配置了什么,下面我們來了解SpringBoot內(nèi)置數(shù)據(jù)源持久化2022-07-07Java中如何將list轉(zhuǎn)為樹形結(jié)構(gòu)
這篇文章主要介紹了Java中如何將list轉(zhuǎn)為樹形結(jié)構(gòu),本文通過示例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-09-09windows定時器配置執(zhí)行java jar文件的方法詳解
這篇文章主要給大家介紹了關(guān)于windows定時器配置執(zhí)行java jar文件的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-11-11Java實現(xiàn)學(xué)生信息管理系統(tǒng)IO版本
這篇文章主要為大家詳細(xì)介紹了Java實現(xiàn)學(xué)生信息管理系統(tǒng)IO版本,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-04-04SpringBoot整合redis實現(xiàn)輸入密碼錯誤限制登錄功能
遇到這樣的需求需要實現(xiàn)一個登錄功能,并且2分鐘之內(nèi)只能輸入5次錯誤密碼,若輸入五次之后還沒有輸入正確密碼,系統(tǒng)將會將該賬號鎖定1小時,這篇文章主要介紹了SpringBoot整合redis并實現(xiàn)輸入密碼錯誤限制登錄功能,需要的朋友可以參考下2024-02-02