Spring中基于XML的面向切面編程(AOP)詳解
一、基于XML的AOP
1.1、打印日志案例
1.1.1、beans.xml中添加aop的約束
<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 https://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop https://www.springframework.org/schema/aop/spring-aop.xsd"> </beans>
1.1.2、定義Bean
package cn.bdqn.domain; public class User { }
package cn.bdqn.service; public interface UserService { // 保存用戶 public void save(User user); // 根據(jù)id查詢用戶 public User queryById(Integer id); // 查詢?nèi)坑脩? public List<User> queryAll(); }
package cn.bdqn.service; public class UserServiceImpl implements UserService{ // 保存用戶 public void save(User user){ } // 根據(jù)id查詢用戶 public User queryById(Integer id){ return new User(); } // 查詢?nèi)坑脩? public List<User> queryAll(){ return new ArrayList<User>(); } }
1.2、定義記錄日志的類【切面】
package cn.bdqn.advice; // 定義記錄日志的類,這個類就封裝了我們所有的公共的代碼 public class Logger { // 該方法的作用是在切入點方法執(zhí)行之前執(zhí)行 public void beforePrintLog(){ System.out.println("開始打印日志啦"); } }
1.3、導(dǎo)入AOP的依賴
<dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjweaver</artifactId> <version>1.9.4</version> </dependency>
1.4、主配置文件中配置AOP
<beans> <!-- 1、注冊UserServiceImpl這個Bean --> <bean id="userService" class="cn.bdqn.service.UserServiceImpl"/> <!-- 2、以下操作都是Spring基于XML的AOP配置步驟 2.1 把通知/增強Bean也需要注冊到Spring容器中 2.2 使用<aop:config/>標(biāo)簽來去聲明開始AOP的配置了 2.3 使用<aop:aspect/>標(biāo)簽來去表示開始配置切面了 可以想一下:既然要配置切面,那切面就是切入點和通知的結(jié)合,所以肯定需要配置切入點和通知這兩部分 id屬性:是給切面提供一個唯一標(biāo)識 ref屬性:是指定通知類bean的Id。 2.4 在<aop:aspect/>標(biāo)簽的內(nèi)部使用對應(yīng)標(biāo)簽來配置通知的類型 前置通知/后置通知/異常通知/最終通知 需求:beforePrintLog方法在切入點方法執(zhí)行之前之前:所以是前置通知 前置通知:<aop:before/> method屬性:用于指定Logger類中哪個方法是前置通知 pointcut屬性:用于指定切入點表達式,該表達式的含義指的是對業(yè)務(wù)層中哪些方法增強 3、切入點表達式的寫法: 關(guān)鍵字:execution(表達式) 表達式: 訪問修飾符 方法返回值 包名1.包名2...類名.方法名(參數(shù)列表) 需求: 我現(xiàn)在就想對UserServiceImpl類中的queryAll方法進行攔截 public java.util.List cn.bdqn.service.UserServiceImpl.queryAll() --> <!-- 2.1 把通知/增強Bean也需要注冊到Spring容器中 --> <bean id="logger" class="cn.bdqn.advice.Logger"/> <!-- 2.2 使用此標(biāo)簽來去聲明開始AOP的配置了--> <aop:config> <!--配置切面 --> <aop:aspect id="loggerAdvice" ref="logger"> <!-- 配置通知的類型,并且建立增強方法和切入點方法的關(guān)聯(lián)--> <aop:before method="beforePrintLog" pointcut="execution(public java.util.List cn.bdqn.service.UserServiceImpl.queryAll())"/> </aop:aspect> </aop:config> </beans>
1.5、測試
@Test public void testUserServiceImpl() throws Exception{ ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml"); UserService userService = (UserService) ac.getBean("userService"); userService.queryAll(); }
1.6、切入點表達式
問題:我們上面的案例經(jīng)過測試發(fā)現(xiàn)確實在調(diào)用業(yè)務(wù)方法之前增加了日志功能,但是問題是僅僅能針對某一個業(yè)務(wù)方法進行增強,而我們的業(yè)務(wù)方法又有可能有很多,所以顯然一個一個的去配置很麻煩,如何更加靈活的去配置呢?這個就需要使用到切入點表達式
? 語法:execution(表達式)
訪問修飾符 方法返回值 包名1.包名2...類名.方法名(參數(shù)列表)
1.6.1、訪問修飾符可以省略
// 完整寫法 public java.util.List cn.bdqn.service.UserServiceImpl.queryAll()) // 標(biāo)準(zhǔn)寫法 java.util.List cn.bdqn.service.UserServiceImpl.queryAll())
1.6.2、返回值可以使用通配符,表示任意返回值
* cn.bdqn.service.UserServiceImpl.queryAll())
1.6.3、包名可以使用通配符表示任意包。有幾級包,就幾個*
* *.*.*.UserServiceImpl.queryAll())
但是對于包來說,連續(xù)的寫3個*,顯然也是麻煩的,那么可以使用“…”表示當(dāng)前包及其子包。
// 表示的是任意包下的只要有UserServiceImpl類都會對queryAll方法進行增強 * *..UserServiceImpl.queryAll())
1.6.4、類名也可以用*
* *..*.queryAll()
1.6.5、方法也可以用*
* *..*.*()
1.6.6、參數(shù)列表
寫法1、可以直接寫數(shù)據(jù)類型:
基本類型直接寫名稱
int、double
引用類型寫包名.類名的方式
java.lang.String、java.util.List
寫法2、可以使用通配符表示任意類型
前提是必須要有參數(shù)。寫法3、使用..
可以使用..表示有無參數(shù)均可,如果有參數(shù)則表示的可以是任意類型
1.6.7、全通配符寫法
* *..*.*(..)
1.6.8、使用最多的寫法
實際中的寫法:切到業(yè)務(wù)層實現(xiàn)類下的所有方法。即:
* com.bdqn.service.impl.*.*(..)
1.7、通知類型的使用
1.7.1、在日志類中新增通知方法
// 定義記錄日志的類,這個類就封裝了我們所有的公共的代碼 public class Logger { // 該方法的作用是在切入點方法執(zhí)行之前執(zhí)行 public void beforePrintLog(){ System.out.println("前置通知(beforePrintLog):開始打印日志啦"); } // 該方法的作用是在切入點方法執(zhí)行之后執(zhí)行 public void afterReturningPrintLog(){ System.out.println("后置通知(afterReturningPrintLog):業(yè)務(wù)方法執(zhí)行完了,日志打印"); } // 該方法的作用是在切入點方法執(zhí)行出錯后執(zhí)行 public void afterThrowingPrintLog(){ System.out.println("異常通知(afterThrowingPrintLog):業(yè)務(wù)方法出現(xiàn)異常了,日志打印"); } // 該方法的作用是在切入點方法執(zhí)行之后不管有沒有錯誤,都最終要執(zhí)行 public void afterPrintLog(){ System.out.println("最終通知(afterPrintLog):業(yè)務(wù)方法不管有沒有異常了,日志打印"); } }
1.7.2、配置AOP
<beans> <!-- 2.1 把通知/增強Bean也需要注冊到Spring容器中 --> <bean id="logger" class="cn.bdqn.advice.Logger"/> <!-- 2.2 使用此標(biāo)簽來去聲明開始AOP的配置了--> <aop:config> <!--配置切面 --> <aop:aspect id="loggerAdvice" ref="logger"> <!-- 配置前置通知:在切入點方法執(zhí)行之前執(zhí)行--> <aop:before method="beforePrintLog" pointcut="execution(* cn.bdqn.service.UserServiceImpl.queryAll())"/> <!-- 后置通知:在切入點方法正常執(zhí)行之后值。它和異常通知永遠只能執(zhí)行一個--> <aop:after-returning method="afterReturningPrintLog" pointcut="execution(* cn.bdqn.service.UserServiceImpl.queryAll())"/> <!--配置異常通知:在切入點方法執(zhí)行產(chǎn)生異常之后執(zhí)行。它和后置通知永遠只能執(zhí)行一個--> <aop:after-throwing method="afterThrowingPrintLog" pointcut="execution(* cn.bdqn.service.UserServiceImpl.queryAll())"/> <!--配置最終通知:無論切入點方法是否正常執(zhí)行它都會在其后面執(zhí)行--> <aop:after method="afterPrintLog" pointcut="execution(* cn.bdqn.service.UserServiceImpl.queryAll())"/> </aop:aspect> </aop:config> </beans>
1.7.3、測試
@Test public void testUserServiceImpl() throws Exception{ ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml"); UserService userService = (UserService) ac.getBean("userService"); userService.queryAll(); } /*** 前置通知(beforePrintLog):開始打印日志啦 查詢?nèi)坑脩魣?zhí)行啦 后置通知(afterReturningPrintLog):業(yè)務(wù)方法執(zhí)行完了,日志打印 最終通知(afterPrintLog):業(yè)務(wù)方法不管有沒有異常了,日志打印 **/
1.8、切入點表達式改進
通過11.7可以發(fā)現(xiàn),我們在配置文件中配置了四種通知類型,其中的pointcut配置的是切入點表達式,發(fā)現(xiàn)是一模一樣的,那么有沒有一種改進寫法呢?可以將表達式抽取出來,將來可以引用。
1.8.1、方式一
<beans> <!-- 1、注冊UserServiceImpl這個Bean --> <bean id="userService" class="cn.bdqn.service.UserServiceImpl"/> <!-- 2、以下操作都是Spring基于XML的AOP配置步驟--> <!-- 2.1 把通知/增強Bean也需要注冊到Spring容器中 --> <bean id="logger" class="cn.bdqn.advice.Logger"/> <!-- 2.2 使用此標(biāo)簽來去聲明開始AOP的配置了--> <aop:config> <!--配置切面 --> <aop:aspect id="loggerAdvice" ref="logger"> <!-- 配置切入點表達式 id屬性用于指定切入點表達式的唯一標(biāo)識。 expression屬性用于指定表達式內(nèi)容 此標(biāo)簽寫在aop:aspect標(biāo)簽內(nèi)部只能當(dāng)前切面使用。 --> <aop:pointcut id="loggerPt" expression="execution(* cn.bdqn.service.UserServiceImpl.queryAll())"/> <!-- 配置前置通知:在切入點方法執(zhí)行之前執(zhí)行--> <aop:before method="beforePrintLog" pointcut-ref="loggerPt"/> <!-- 后置通知:在切入點方法正常執(zhí)行之后值。它和異常通知永遠只能執(zhí)行一個--> <aop:after-returning method="afterReturningPrintLog" pointcut-ref="loggerPt"/> <!--配置異常通知:在切入點方法執(zhí)行產(chǎn)生異常之后執(zhí)行。它和后置通知永遠只能執(zhí)行一個--> <aop:after-throwing method="afterThrowingPrintLog" pointcut-ref="loggerPt"/> <!--配置最終通知:無論切入點方法是否正常執(zhí)行它都會在其后面執(zhí)行--> <aop:after method="afterPrintLog" pointcut-ref="loggerPt"/> </aop:aspect> </aop:config> </beans>
1.8.2、方式二
對于方式一,我們將aop:pointcut標(biāo)簽寫在了aop:aspect里面,這樣的話這切入點表達式只能被當(dāng)前的切面使用,而如果其他切面想使用就使用不到了,所以我們可以把這個切入點表示再定義到外面。
<beans> <bean id="userService" class="cn.bdqn.service.UserServiceImpl"/> <!-- 2、以下操作都是Spring基于XML的AOP配置步驟--> <!-- 2.1 把通知/增強Bean也需要注冊到Spring容器中 --> <bean id="logger" class="cn.bdqn.advice.Logger"/> <!-- 2.2 使用此標(biāo)簽來去聲明開始AOP的配置了--> <aop:config> <!-- 配置切入點表達式 id屬性用于指定切入點表達式的唯一標(biāo)識。 expression屬性用于指定表達式內(nèi)容 此標(biāo)簽寫在aop:aspect標(biāo)簽外面,那么所有的切面都可以使用。 --> <aop:pointcut id="loggerPt" expression="execution(* cn.bdqn.service.UserServiceImpl.queryAll())"/> <!--配置切面 --> <aop:aspect id="loggerAdvice" ref="logger"> <!-- 配置前置通知:在切入點方法執(zhí)行之前執(zhí)行--> <aop:before method="beforePrintLog" pointcut-ref="loggerPt"/> <!-- 后置通知:在切入點方法正常執(zhí)行之后值。它和異常通知永遠只能執(zhí)行一個--> <aop:after-returning method="afterReturningPrintLog" pointcut-ref="loggerPt"/> <!--配置異常通知:在切入點方法執(zhí)行產(chǎn)生異常之后執(zhí)行。它和后置通知永遠只能執(zhí)行一個--> <aop:after-throwing method="afterThrowingPrintLog" pointcut-ref="loggerPt"/> <!--配置最終通知:無論切入點方法是否正常執(zhí)行它都會在其后面執(zhí)行--> <aop:after method="afterPrintLog" pointcut-ref="loggerPt"/> </aop:aspect> </aop:config> </beans>
1.9、環(huán)繞通知
1.9.1、在日志記錄類中新增環(huán)繞通知
public class Logger { // 環(huán)繞通知 public void aroundPrintLog(){ System.out.println("環(huán)繞通知....aroundPrintLog....."); } }
1.9.2、AOP配置環(huán)繞通知
<beans> <!-- 1、注冊UserServiceImpl這個Bean --> <bean id="userService" class="cn.bdqn.service.UserServiceImpl"/> <!-- 2、以下操作都是Spring基于XML的AOP配置步驟--> <!-- 2.1 把通知/增強Bean也需要注冊到Spring容器中 --> <bean id="logger" class="cn.bdqn.advice.Logger"/> <!-- 2.2 使用此標(biāo)簽來去聲明開始AOP的配置了--> <aop:config> <!-- 配置切入點表達式 --> <aop:pointcut id="loggerPt" expression="execution(* cn.bdqn.service.UserServiceImpl.queryAll())"/> <!--配置切面 --> <aop:aspect id="loggerAdvice" ref="logger"> <!-- 環(huán)繞通知--> <aop:around method="aroundPrintLog" pointcut-ref="loggerPt"/> </aop:aspect> </aop:config> </beans>
1.9.3、測試1
@Test public void testUserServiceImpl() throws Exception{ ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml"); UserService userService = (UserService) ac.getBean("userService"); userService.queryAll(); } /** 環(huán)繞通知....aroundPrintLog..... 發(fā)現(xiàn):僅僅打印了環(huán)繞通知的代碼。當(dāng)我們配置了環(huán)繞通知之后,切入點方法沒有執(zhí)行,而通知方法執(zhí)行了 */
1.9.4、解決
Spring框架為我們提供了一個接口:ProceedingJoinPoint。該接口有一個方法proceed(),此方法就相當(dāng)于明確調(diào)用切入點方法。該接口可以作為環(huán)繞通知的方法參數(shù),在程序執(zhí)行時,spring框架會為我們提供該接口的實現(xiàn)類供我們使用。
public class Logger { // 環(huán)繞通知 public Object aroundPrintLog(ProceedingJoinPoint pjp){ Object result = null; try{ Object[] args = pjp.getArgs(); System.out.println(pjp.getSignature().getName()); System.out.println("前置"); result = pjp.proceed(args); System.out.println("后置"); return result; }catch (Throwable t){ System.out.println("異常"); throw new RuntimeException(t); }finally { System.out.println("最終"); } } } /** 環(huán)繞通知:它是spring框架為我們提供的一種可以在代碼中手動控制增強方法何時執(zhí)行的方式。 */
以上就是Spring中基于XML的面向切面編程(AOP)詳解的詳細內(nèi)容,更多關(guān)于Spring 于XML面向切面編程的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
從log4j切換到logback后項目無法啟動的問題及解決方法
這篇文章主要介紹了從log4j切換到logback后項目無法啟動的問題及解決方法,本文給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-01-01Java JDBC連接Kerberos認證的HIVE和Impala方式
本文主要介紹了HiveJDBC和ImpalaJDBC的使用方法,包括版本對應(yīng)、Maven安裝、主機名配置、端口開通、JDBC連接和Kerberos認證等2025-02-02Java多線程+鎖機制實現(xiàn)簡單模擬搶票的項目實踐
鎖是一種同步機制,用于控制對共享資源的訪問,在線程獲取到鎖對象后,可以執(zhí)行搶票操作,本文主要介紹了Java多線程+鎖機制實現(xiàn)簡單模擬搶票的項目實踐,具有一定的參考價值,感興趣的可以了解一下2024-02-02使用HttpSessionListener監(jiān)聽器實戰(zhàn)
這篇文章主要介紹了使用HttpSessionListener監(jiān)聽器實戰(zhàn),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-03-03詳解 Java繼承關(guān)系下的構(gòu)造方法調(diào)用
這篇文章主要介紹了詳解 Java繼承關(guān)系下的構(gòu)造方法調(diào)用的相關(guān)資料,希望通過本文能幫助到大家,讓大家理解掌握這部分內(nèi)容,需要的朋友可以參考下2017-10-10