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