SpringBoot框架aop切面的execution表達(dá)式解讀
SpringBoot aop切面execution表達(dá)式
Aspectj切入點(diǎn)語(yǔ)法定義在使用spring框架配置AOP的時(shí)候,不管是通過(guò)XML配置文件還是注解的方式都需要定pointcut"切入點(diǎn)"
例如:
定義切入點(diǎn)表達(dá)式 execution (* com.nandao.demo.controller.*.*.*(..))
/** * 監(jiān)控controller層的接口 */ @Pointcut("execution(* com.nandao.demo.controller.*.*.*(..))") private void pointCut(){}
execution()是最常用的切點(diǎn)函數(shù),其語(yǔ)法如下所示:
整個(gè)表達(dá)式可以分為五個(gè)部分
1、execution(): 表達(dá)式主體,可以掃描控制層的接口、某個(gè)注解、或者其他需要掃描的類(lèi)。
2、第一個(gè)*號(hào):表示返回類(lèi)型,*號(hào)表示所有的類(lèi)型,比如public,protect,private等。
3、包名:表示需要攔截的包名,后面的兩個(gè)句點(diǎn)表示當(dāng)前包和當(dāng)前包的所有子包,com.demo.service.impl包、子孫包下所有類(lèi)的方法。
4、第二個(gè)*號(hào):表示子包名,*號(hào)表示所有子包。
5、第三個(gè)*號(hào):表示類(lèi)名,*號(hào)表示所有子包下的類(lèi)。
6、*(..):最后這個(gè)星號(hào)表示方法名,*號(hào)表示所有的方法,后面括弧里面表示方法的參數(shù),兩個(gè)句點(diǎn)表示任何參數(shù)。
springboot aop自定義切面編程
本文結(jié)合springboot,實(shí)現(xiàn)切面編程。
- @Before 前置通知:在某連接點(diǎn)(JoinPoint)之前執(zhí)行的通知, 但這個(gè)通知不能阻止連接點(diǎn)前的執(zhí)行。
- @After 后通知:當(dāng)某連接點(diǎn)退出的時(shí)候執(zhí)行的通知 (不論是正常返回還是異常退出)。
- @AfterReturning 返回后通知 :在某連接點(diǎn)正常完成后執(zhí)行的通知, 不包括拋出異常的情況。
- @Around 環(huán)繞通知 :包圍一個(gè)連接點(diǎn)的通知,類(lèi)似Web中Servlet ,規(guī)范中的Filter的doFilter方法。可以在方法的調(diào)用前后完成 自定義的行為,也可以選擇不執(zhí)行。
- @AfterThrowing 拋出異常后通知:在方法拋出異常退出時(shí)執(zhí)行的通知。
1.首先pom文件引入aop依賴
<!-- spring-boot的aop切面服務(wù) --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-aop</artifactId> </dependency>
2.新建兩個(gè)類(lèi)
- 一個(gè)實(shí)現(xiàn)切面的處理類(lèi)
- 一個(gè)定義注解
首先是定義注解:
@Documented @Retention(RetentionPolicy.RUNTIME) @Target(ElementType.METHOD) public @interface TestAnnotation { }
然后是:切面處理類(lèi)
@Order(2) @Aspect //定義一個(gè)切面 @Configuration public class TestAspect { // 定義切點(diǎn)Pointcut //第一個(gè)注解:表示掃描的包或類(lèi), 第一個(gè)*可以寫(xiě)具體的類(lèi),第二個(gè)*是方法,括號(hào)里是傳參 //第二個(gè)注解表示定義的注解, 哪個(gè)方法需要切面,就在方法上加上 @TestAnnotation @Pointcut("execution(public * com.example.demo.controller.*.*(..)) && @annotation(com.example.demo.aop.annotation.TestAnnotation)") public void executeService() { } //執(zhí)行方法之前,進(jìn)入切面 @Before(value = "executeService()") public void doBeforeAdvice(JoinPoint joinPoint) { } //在進(jìn)去方法的同時(shí),進(jìn)入切面 @Around("executeService()") public Object doAroundAdvice(ProceedingJoinPoint pjp) throws Throwable { RequestAttributes ra = RequestContextHolder.getRequestAttributes(); ServletRequestAttributes sra = (ServletRequestAttributes) ra; HttpServletResponse response = sra.getResponse(); HttpServletRequest request = sra.getRequest(); String uri = request.getRequestURI(); Object result = pjp.proceed(); return result; } //在執(zhí)行完方法后,進(jìn)入切面,并返回方法的結(jié)果值,returning定義的參數(shù)必須和Object后的形參一致 @AfterReturning(value = "executeService()", returning = "result") public void doAfterAdvice(JoinPoint joinPoint, Object result) throws Throwable { RequestAttributes ra = RequestContextHolder.getRequestAttributes(); ServletRequestAttributes sra = (ServletRequestAttributes) ra; HttpServletResponse response = sra.getResponse(); if (response.getStatus() == 200) { HttpServletRequest request = sra.getRequest(); String uri = request.getRequestURI(); HttpSession session = request.getSession(true); Gson gson = new Gson(); JSONObject jsonObject = JSON.parseObject(gson.toJson(result)); } } }
具體方法 的注解已經(jīng)放到代碼中,
Gson 的pom依賴:
<!-- https://mvnrepository.com/artifact/com.google.code.gson/gson --> <dependency> <groupId>com.google.code.gson</groupId> <artifactId>gson</artifactId> <version>2.8.5</version> </dependency>
注意:
假如定義了一個(gè)切面: 即一個(gè)方法只被一個(gè)aspect類(lèi)攔截,aspect類(lèi)內(nèi)部的 advice 將按照以下的順序進(jìn)行執(zhí)行:
若同一個(gè)方法被多個(gè)Aspect類(lèi)攔截,執(zhí)行順序是不一定的。
想要有一定的順序,可以在處理類(lèi)中加入@Order 注解;
執(zhí)行順序如下:
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
淺談SpringCloud?Alibaba和SpringCloud的區(qū)別
這篇文章主要介紹了淺談SpringCloud?Alibaba和SpringCloud的區(qū)別,Spring?Cloud?Netflix框架也屬于Spring?Cloud,但是Netflix并不是由spring來(lái)進(jìn)行開(kāi)發(fā)的,需要的朋友可以參考下2023-05-05spring?boot項(xiàng)目自定義參數(shù)校驗(yàn)規(guī)則示例詳解
這篇文章主要介紹了spring boot項(xiàng)目如何自定義參數(shù)校驗(yàn)規(guī)則,自定義校驗(yàn)規(guī)則和自帶的規(guī)則實(shí)現(xiàn)方式一樣,先自定義一個(gè)注解,然后指定校驗(yàn)類(lèi),在校驗(yàn)類(lèi)里實(shí)現(xiàn)具體的校驗(yàn)規(guī)則,本文結(jié)合示例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2023-07-07Java基礎(chǔ)之動(dòng)態(tài)代理Cglib詳解
這篇文章主要介紹了Java基礎(chǔ)之動(dòng)態(tài)代理Cglib詳解,文中有非常詳細(xì)的代碼示例,對(duì)正在學(xué)習(xí)java基礎(chǔ)的小伙伴們有非常好的幫助,需要的朋友可以參考下2021-05-05java實(shí)現(xiàn)PDF轉(zhuǎn)圖片的方法
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)PDF轉(zhuǎn)圖片的方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-07-07深入解析JVM之內(nèi)存結(jié)構(gòu)及字符串常量池(推薦)
Java作為一種平臺(tái)無(wú)關(guān)性的語(yǔ)言,其主要依靠于Java虛擬機(jī)——JVM,接下來(lái)通過(guò)本文給大家介紹JVM之內(nèi)存結(jié)構(gòu)及字符串常量池的相關(guān)知識(shí),需要的朋友可以參考下2020-07-07java編程調(diào)用存儲(chǔ)過(guò)程中得到新增記錄id號(hào)的實(shí)現(xiàn)方法
這篇文章主要介紹了java編程調(diào)用存儲(chǔ)過(guò)程中得到新增記錄id號(hào)的實(shí)現(xiàn)方法,涉及Java數(shù)據(jù)庫(kù)操作中存儲(chǔ)過(guò)程的相關(guān)使用技巧,需要的朋友可以參考下2015-10-10