Springboot AOP開發(fā)教程
Springboot AOP開發(fā)
簡介
AOP(Aspect Oriented Programming)意為:面向切面編程,通過預編譯方式和運行期動態(tài)代理實現(xiàn)程序功能的統(tǒng)一維護的一種技術。AOP是OOP的延續(xù),是軟件開發(fā)中的一個熱點,也是Spring框架中的一個重要內(nèi)容,是函數(shù)式編程的一種衍生范型。利用AOP可以對業(yè)務邏輯的各個部分進行隔離,從而使得業(yè)務邏輯各部分之間的耦合度降低,提高程序的可重用性,同時提高了開發(fā)的效率。
一 AOP概述
AOP,即面向切面編程,簡言之,面向方法編程。
針對方法,在方法的執(zhí)行前或執(zhí)行后使用,用于增強方法,或拓展。
二 AOP開發(fā)
1.引入 spring-boot-starter-aop
在SpringBoot項目的pom文件中,引入 spring-boot-starter-aop依賴。
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-aop</artifactId> </dependency>
2.示例:計算方法執(zhí)行時間
1.創(chuàng)建實體類,通過注解來申明該類的類型,并將該類交給Spring的IOC容器來管理。
通過注解 @Aspect
申明這是一個AOP類
通過 @Component
將其交給IOC容器管理
@Aspect @Component public class TimeAspect { //code }
2.創(chuàng)建方法并且實現(xiàn)
@Aspect @Component @Slf4j public class TimeAspect { // 針對 com.shawn.springboot03.service 包下所有的方法進行編程, // * com.shawn.springboot03.service.*.*(..)) 為切入點表達式 @Around("execution(* com.shawn.springboot03.service.*.*(..))") public Object recordTime(ProceedingJoinPoint proceedingJoinPoint) throws Throwable { long startTime=System.currentTimeMillis(); // 切面對象,執(zhí)行具體的業(yè)務方法 Object object = proceedingJoinPoint.proceed(); long endTime=System.currentTimeMillis(); log.info("方法耗時為{}ms",endTime-startTime); return object; } }
通過以上方法,當用戶調(diào)用 service 層接口的任一方法時都會計算方法的運行時間。
3.AOP編程的優(yōu)點:
- 代碼無侵入:無需修改原始方法
- 減少代碼重復,提高開發(fā)效率:只需編寫一次
- 維護方便:根據(jù)業(yè)務需求,調(diào)整切入點表達式即可
三 AOP詳解
1.AOP核心概念
1.連接點(JoinPoint):連接點指的是可以被AOP控制的方法,以及方法執(zhí)行時的相關信息。
2.通知(Advice):Advice指的是被抽取出來的共性功能,即重復的那部分邏輯。
3.切入點(PointCut):匹配連接點的條件,通知僅會在切入點方法執(zhí)行時被應用
4.切面(Aspcet):描述通知與切入點的關系
5.目標對象(Target):通知所應用的對象
2.AOP通知類型
- @Around:環(huán)繞通知,此注解標注的方法在目標方法前后都會執(zhí)行
- @Before:前置通知,此注解標注的方法僅在方法執(zhí)行前被執(zhí)行
- @After:后置通知,此注解在方法執(zhí)行完成后執(zhí)行,不論是否拋出異常
- @AfterReturning:返回后通知,此注解標注的方法在目標方法后被執(zhí)行,有異常不通知
- @AfterThrowing:異常后通知,此注解的通知方法發(fā)生異常后執(zhí)行
3.各通知類型演示
創(chuàng)建一個 TestAspect 類,用于演示各種通知類型
@Around 通知類型
介紹:在方法前后均執(zhí)行
切入點表達式格式: 返回值 包名.方法名(形參)
其中 *
代表全部,..
代表任意多的參數(shù)
切入點表達式示例說明:
* com.shawn.test.server.*(..)
表示 任意返回值的 com.shawn.test.server
包下任意返回值,任意多參數(shù)的全部方法。
@Aspect @Component @Slf4j public class TestAspect { @Around("execution(* com.shawn.springboot03.service.impl.DeptServiceImpl.*(..))") public Object around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable { //方法執(zhí)行前的業(yè)務邏輯 //code.. //執(zhí)行原始方法(即連接點),并且接收原始方法的返回值 Object proceed = proceedingJoinPoint.proceed(); //方法執(zhí)行后的業(yè)務邏輯 //code.. //返回原始方法的返回值 return proceed; } }
ProceedingJoinPoint
是一個接口類,指代程序執(zhí)行過程中的一個特定點,其中包含了原始方法的全部內(nèi)容,包括方法名,參數(shù)值等。
在使用 @Around
通知類型時,需要將該對象作為參數(shù)
傳遞進來,用于后續(xù)執(zhí)行原始方法或獲取原始方法的其他信息。
ProceedingJoinPoint
僅能作用于 @Around
類型的通知上。
2.@Before 通知類型
介紹:此注解標注的方法僅在原始方法執(zhí)行前執(zhí)行
@Aspect @Component @Slf4j public class TestAspect { @Before("execution(* com.shawn.springboot03.service.*.*(..))") public void before(){ // 方法執(zhí)行前的業(yè)務邏輯 //code.. } }
3.@After 通知類型
介紹:此注解標注的方法,僅在原始方法執(zhí)行后執(zhí)行,且不論原始方法是否執(zhí)行,該通知都會執(zhí)行
@Aspect @Component @Slf4j public class TestAspect { @After("execution(* com.shawn.springboot03.service.*.*(..))") public void after(){ // 方法執(zhí)行后的業(yè)務邏輯 //code.. } }
4.@AfterReturning 通知類型
介紹:此注解標注的方法,僅在原始方法執(zhí)行完成后通知,即當原始方法執(zhí)行發(fā)生異常時,@AfterReturning
通知不會被執(zhí)行。
@Aspect @Component @Slf4j public class TestAspect { @AfterReturning("execution(* com.shawn.springboot03.service.*.*(..))") public void afterReturning(){ //方法執(zhí)行完成的業(yè)務邏輯 //code.. } }
5.@AfterThrowing通知類型
介紹:次注解標注的方法,僅在原始方法執(zhí)行過程中發(fā)生了異常,才會執(zhí)行。
@Aspect @Component @Slf4j public class TestAspect { /** * 聲明一個空的方法體來定義切入點表達式 * 使用 @Pointcut 注解來定義切入點表達式 */ @Pointcut("execution(* com.shawn.springboot03.service.impl.DeptServiceImpl.*(..))") private void point(){}; @After("point()") public void after(){ // 方法執(zhí)行后的業(yè)務邏輯 //code.. } }
4.定義切入點
以上示例中的切入點表達式均相似,可以利用封裝的思想,將切入點表達式全部抽取出來,在需要的時候直接使用即可。
想要實現(xiàn)以上需求,則需要自定義切入點表達式
通過 @Pointcut
注解,來定義切入點表達式,然后在需要編寫切入點表達式的地方調(diào)用即可。
@Aspect @Component public class TestAspect { @Pointcut("@annotation(com.shawn.springboot03.annotation.OperationLog)") private void point(){}; @AfterThrowing("point()") public void afterThrowing(){ //方法執(zhí)行時拋出異常的業(yè)務邏輯 //code.. } }
注意:如果自定義的切入點訪問修飾符為 public
,則該表達式還可以在其他切面類中被引用。具體使用可根據(jù)實際業(yè)務情況決定。
定義切入點總共有兩種辦法,一種是通過方法名來定義切入點,還可以通過注解來定義切入點。
示例:以下示例使用自定義注解作為切入點,使用了以下自定義注解的方法會執(zhí)行通知。
@Aspect @Component @Slf4j @Order(3) public class TestAspect { @Pointcut("execution(* com.shawn.springboot03.service.impl.DeptServiceImpl.*(..))") private void point(){}; @After("point()") public void after(){ // 方法執(zhí)行后的業(yè)務邏輯 //code.. } }
5.通知順序
當有多個切入點都匹配到了目標方法,目標方法運行時,多個通知都會被執(zhí)行。
某些情況下則需要控制通知的順序,可以在切面類上使用 @Order
注解來控制順序
@Aspect @Component @Slf4j @Order(3) public class TestAspect { @Pointcut("execution(* com.shawn.springboot03.service.impl.DeptServiceImpl.*(..))") private void point(){}; @After("point()") public void after(){ // 方法執(zhí)行后的業(yè)務邏輯 //code.. } }
注意:
@Order(number)
注解中,數(shù)字越小的越先運行,越大的越后運行。默認情況下,按照切面類的類名排序順序執(zhí)行。
四 切入點表達式
1.execution
execution 主要根據(jù)方法的返回值,包名類名,方法名,方法參數(shù)等信息來匹配,語法為:
execution( 訪問修飾符? 返回值 包名.類名.?方法名(方法參數(shù)) throw 異常? )
其中帶 ? 的表示可以省略的部分
- 訪問修飾符:可省略(比如:public,protected)
- 包名.類名:可省略
- throws 異常:可省略(注意是方法上聲明拋出的異常,不是實際拋出的異常)
以下是一個完整示例:
@Before ("execution(public void com.itheima.service.impl.DeptserviceImpl.delete (java.lang.Integer)) throws Exception") private void point(){};
省略后的示例:
@Before ("execution(void delete (java.lang.Integer))") private void point(){};
此時,所有 void delete (java.lang.Integer)
方法都將被匹配到。
注意: * 用于描述匹配單個, … 可以用來匹配多個連續(xù),?表示可省略
2.@annotation
@annotation
切入點表達式,用于匹配標識有特定注解的方法。
示例:
@Pointcut("@annotation(com.shawn.annotation.OperationLog)") private void point(){};
五 連接點
在Spring中使用JoinPoint
抽象了連接點,使用它可以獲取方法執(zhí)行時的相關信息,入目標類名,方法名,方法參數(shù)等。
1.對于 @Around
通知,獲取連接點信息只能使用 ProceedingJoinPoint
@Around("point()") public Object around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable { //獲取目標類名 String name = proceedingJoinPoint.getTarget().getClass().getName(); log.info("獲取目標類名:"+name); //獲取方法執(zhí)行參數(shù) Object[] args = proceedingJoinPoint.getArgs(); log.info("獲取方法執(zhí)行參數(shù):"+JSON.toJSONString(args)); //獲取目標方法名 String methodName = proceedingJoinPoint.getSignature().getName(); log.info("獲取目標方法名:"+methodName); //執(zhí)行原始方法(即連接點),并且接收原始方法的返回值 Object proceed = proceedingJoinPoint.proceed(); //獲取方法的返回值 log.info("獲取方法的返回值:"+JSON.toJSONString(proceed)); //返回原始方法的返回值 return proceed; }
proceedingJoinPoint.proceed()得到目標方法的結果再返回,此處可以對目標方法執(zhí)行的結果進行篡改。
2.對于其他四種通知,獲取連接點信息只能使用 JoinPoint
, 它是 ProceedingJoinPoint
的父類。
@Before("point()") public void before(JoinPoint joinPoint){ // 獲取目標方法的類名 String name = joinPoint.getTarget().getClass().getName(); log.info("獲取目標方法的類名:"+name); String methodName = joinPoint.getSignature().getName(); log.info("獲取目標方法的方法名:"+methodName); //獲取目標方法的運行參數(shù) Object[] args = joinPoint.getArgs(); log.info("獲取目標方法的參數(shù):"+JSON.toJSONString(args)); }
3.除@Around 通知外,還可以獲取到方法執(zhí)行結果的通知類型為 @AfterReturning,@AfterReturning 在方法執(zhí)行完成并且無異常時通知。
在定義@AfterReturning通知類型時,使用 pointcut 屬性定義切入點,returning屬性定義返回值對象,然后在方法參數(shù)中傳入即可。
@AfterReturning(pointcut = "point()",returning = "object") public void afterReturning(JoinPoint joinPoint, Object object){ // 獲取目標方法的類名 String name = joinPoint.getTarget().getClass().getName(); log.info("獲取目標方法的類名:"+name); String methodName = joinPoint.getSignature().getName(); log.info("獲取目標方法的方法名:"+methodName); //獲取目標方法的運行參數(shù) Object[] args = joinPoint.getArgs(); log.info("獲取目標方法的參數(shù):"+JSON.toJSONString(args)); //獲取方法返回值 log.info("獲取方法返回值:"+JSON.toJSONString(object)); }
到此這篇關于Springboot AOP開發(fā)的文章就介紹到這了,更多相關Springboot AOP開發(fā)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
IntelliJ?IDEA?2020.2?全家桶及以下版本激活工具大全【喜訊】
這篇文章主要介紹了IntelliJ?IDEA?2020.2?全家桶及以下版本激活工具大全【喜訊】,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-09-09Spring實現(xiàn)內(nèi)置監(jiān)聽器
這篇文章主要介紹了Spring 實現(xiàn)自定義監(jiān)聽器案例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧,希望能給你帶來幫助2021-07-07java使double保留兩位小數(shù)的多方法 java保留兩位小數(shù)
這篇文章主要介紹了java使double類型保留兩位小數(shù)的方法,大家參考使用吧2014-01-01javax.validation包里@NotNull等注解的使用方式
這篇文章主要介紹了javax.validation包里@NotNull等注解的使用方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-01-01