Spring中的AOP操作你瞭解嗎
一、AOP操作術語
1. 連接點
類裡面哪些方法可以被增強,這些可以被增強的方法就稱為連接點。
2. 切入點
實際被真正增強的方法,稱為切入點。
3. 通知(增強)
(1)實際增強的邏輯部分稱為通知(增強)
(2)通知有如下多種類型,如下:
- 前置通知
- 後置通知
- 環繞通知
- 異常通知
- 最終通知(finally)
4. 切面
切面是一個動作,指的是將通知應用到切入點的過程,就叫做切面。
二、AOP操作
Spring 框架一般都是基於 AspectJ 實現 AOP 操作。AspectJ 不是 Spring 組成部分,而是獨立 AOP 框架,一般把 AspectJ 和 Spirng 框架一起使用,進行 AOP 操作 。
基於AspectJ實現AOP操作有如下兩種方式:
- 基於 xml 配置文件實現
- 基於註解方式實現(使用)
2.1 切入點表達式
切入點表達式作用:知道對哪個類裡面的哪個方法進行增強 ,語法結構如下所示:
execution([權限修飾符] [返回類型] [類全路徑] [方法名稱]([參數列表]) )
如下圖所示:
編寫切點示例如下:
1. 如對com.wyf.spring5.Book類中的add進行增強,則切入點表達式如下:
execution(* com.wyf.spring5.Book.add(..))
2. 如對com.wyf.spring5.Book類中所有的方法增強:
execution(* com.wyf.spring5.Book.*(..))
3. 如對 com.wyf.spring5包中的所有類,類中的所有方法都進行加強
execution(* com.wyf.spring5.*.* (..))
2.2 AOP操作(AspectJ 註解方式)
1) 首先我們創建一個類,並添加一個方法:
/** * 被增強類 */public class User { public void add(){ System.out.println("add*****"); }}/** * 被增強類 */ public class User { public void add(){ System.out.println("add*****"); } }
2)接著我們創建增強類(編寫增強邏輯)
在增強類中創建方法,讓不同的方法代表不同的通知類型
/** * 增強類 */ public class UserProxy { /** * 前置通知邏輯 */ public void before(){ System.out.println("before*****"); } }
3)進行通知的配置
在spring配置文件中,開啟註解掃描。(采用java配置類或xml配置文件實現)
<?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context" xmlns:aop="http://www.springframework.org/schema/aop" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd"> <!-- 開啟註解掃描 --> <context:component-scan base-package="com.wyf.aopanno"></context:component-scan> <!-- 開啟Aspect 生成代理對象 --> <aop:aspectj-autoproxy></aop:aspectj-autoproxy> </beans><?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context" xmlns:aop="http://www.springframework.org/schema/aop" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd"> <!-- 開啟註解掃描 --> <context:component-scan base-package="com.wyf.aopanno"></context:component-scan> <!-- 開啟Aspect 生成代理對象 --> <aop:aspectj-autoproxy></aop:aspectj-autoproxy> </beans>
- 使用註解創建User和UserProxy對象(類上面添加創建對象註解,此處為 @Component)
- 在增強類上面添加註解@Aspect
/** * 被增強類 */ @Component public class User { public void add(){ System.out.println("add*****"); } } /** * 增強類 */ @Component @Aspect //生成代理對象 public class UserProxy { /** * 前置通知邏輯 */ public void before(){ System.out.println("before*****"); } }
在 spring 配置文件中開啟生成代理對象
<!-- 開啟Aspect 生成代理對象 --> <aop:aspectj-autoproxy></aop:aspectj-autoproxy>
開啟Aspect生成代理對象,相當掃描帶有@Aspect註解的類,並生成該對象的一個代理對象。
4)配置不同類型通知
在增強類的裡面,在作為通知方法上面添加通知類型註解,使用切入點表達式配置。
/** * 增強類 */ @Component @Aspect //生成代理對象 public class UserProxy { /** * 前置通知邏輯 */ //@Before註解表示其作為前置通知 //切入點表達式指定該通知為哪個類的哪個方法進行增強。 @Before(value = "execution(* com.wyf.aopanno.User.add(..))") public void before(){ System.out.println("before*****"); } }
測試代碼如下:
@Test public void TestAop(){ //1. 加載spring 配置文件 ApplicationContext context = new ClassPathXmlApplicationContext("beanAop1.xml"); //得到對象 User user = context.getBean("user",User.class); user.add(); }
執行結果:
前文隻給出瞭前置通知@Before的代碼,下面給出所有5種通知的示例代碼,對上例的所有通知情況進行補全。
/** * 增強類 */ @Component @Aspect //生成代理對象 public class UserProxy { /** * 前置通知邏輯 */ //@Before註解表示其作為前置通知 //切入點表達式指定該通知為哪個類的哪個方法進行增強。 @Before(value = "execution(* com.wyf.aopanno.User.add(..))") public void before(){ System.out.println("before*****"); } /** * 後置通知(返回通知) */ @AfterReturning(value = "execution(* com.wyf.aopanno.User.add(..))") public void afterReturning(){ System.out.println("afterReturning*****"); } /** * 最終通知,有異常也會執行 */ @After(value = "execution(* com.wyf.aopanno.User.add(..))") public void after(){ System.out.println("after*****"); } /** * 異常通知 */ @AfterThrowing(value = "execution(* com.wyf.aopanno.User.add(..))") public void afterthrowing(){ System.out.println("afterthrowing*****"); } /** * 環繞通知,在方法之前和之後均通知 */ @Around(value = "execution(* com.wyf.aopanno.User.add(..))") public void around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable { System.out.println("環繞之前*****"); //執行被增強的方法 proceedingJoinPoint.proceed(); System.out.println("環繞之後*****"); }
代碼執行結果如下:
其中,after是在被增強方法之後執行,而afterreturning是在方法返回值之後執行。
2.3 相同切入點的抽取
在上例種,我們編寫瞭被增強方放的5種通知,分別為前置通知、後置通知、異常通知、最終通知以及環繞通知。我們在編寫增強類時,在通知註解中通過切入點表達式,指明瞭通知要對哪個類的哪個方法進行增強。但是,我們會發現,我們對同一方法進行增強時,其切入點表達式是相同的,為瞭避免重復,我們可以相同切入點進行抽取。
下例中,我們將相同切入點用@Pointcut進行瞭抽取,代碼如下所示:
//對相同切入點進行抽取 @Pointcut(value = "execution(* com.wyf.aopanno.User.add(..))") public void pointdemo(){ }
當我們在其他地方要用該切入點表達式時,直接用其方法名稱就可以。
如下所示:
@Component @Aspect @Order(2) public class PersonProxy { //前置通知 @Before(value = "execution(* com.wyf.aopanno.User.add(..))") public void before(){ System.out.println("Person Before****"); } }
2.4 多個增強類對同一個方法進行增強,設置增強類優先級
假如我們現在又有一個增強類,其中也包含一個before()方法,也對被增強類User的add()方法進行前置增強。那麼我們如何設置其增強的順序呢?
我們通過在增強類上面添加註解 @Order(數字類型值),數字類型值越小優先級越高 ,來保證增強的順尋,代碼如下:
@Component @Aspect @Order(2) public class PersonProxy { //前置通知 @Before(value = "execution(* com.wyf.aopanno.User.add(..))") public void before(){ System.out.println("Person Before****"); } }
三、結束
本文主要介紹瞭AOP關鍵子以及基於註解的AOP操作,通過一個實例,介紹瞭AOP的前置通知、後置通知、異常通知、最終通知以及環繞通知。
本篇文章就到這裡瞭,希望能夠給你帶來幫助,也希望您能夠多多關註WalkonNet的更多內容!
推薦閱讀:
- Java SpringAOP技術之註解方式詳解
- 一篇文章帶你瞭解Spring AOP 的註解
- Spring框架學習之AOP詳解
- 詳解Java SpringAOP切面類
- Spring-AOP @AspectJ進階之如何綁定代理對象