一、基于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);
// 根據id查詢用戶
public User queryById(Integer id);
// 查詢全部用戶
public List<User> queryAll();
}
package cn.bdqn.service;
public class UserServiceImpl implements UserService{
// 保存用戶
public void save(User user){
}
// 根據id查詢用戶
public User queryById(Integer id){
return new User();
}
// 查詢全部用戶
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、導入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/>標簽來去聲明開始AOP的配置了
2.3 使用<aop:aspect/>標簽來去表示開始配置切面了
可以想一下:既然要配置切面,那切面就是切入點和通知的結合,所以肯定需要配置切入點和通知這兩部分
id屬性:是給切面提供一個唯一標識
ref屬性:是指定通知類bean的Id。
2.4 在<aop:aspect/>標簽的內部使用對應標簽來配置通知的類型
前置通知/后置通知/異常通知/最終通知
需求:beforePrintLog方法在切入點方法執(zhí)行之前之前:所以是前置通知
前置通知:<aop:before/>
method屬性:用于指定Logger類中哪個方法是前置通知
pointcut屬性:用于指定切入點表達式,該表達式的含義指的是對業(yè)務層中哪些方法增強
3、切入點表達式的寫法:
關鍵字:execution(表達式)
表達式:
訪問修飾符 方法返回值 包名1.包名2...類名.方法名(參數列表)
需求:
我現在就想對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 使用此標簽來去聲明開始AOP的配置了-->
<aop:config>
<!--配置切面 -->
<aop:aspect id="loggerAdvice" ref="logger">
<!-- 配置通知的類型,并且建立增強方法和切入點方法的關聯(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、切入點表達式
? 問題:我們上面的案例經過測試發(fā)現確實在調用業(yè)務方法之前增加了日志功能,但是問題是僅僅能針對某一個業(yè)務方法進行增強,而我們的業(yè)務方法又有可能有很多,所以顯然一個一個的去配置很麻煩,如何更加靈活的去配置呢?這個就需要使用到切入點表達式
? 語法:execution(表達式)
訪問修飾符 方法返回值 包名1.包名2...類名.方法名(參數列表)
1.6.1、訪問修飾符可以省略
// 完整寫法
public java.util.List cn.bdqn.service.UserServiceImpl.queryAll())
// 標準寫法
java.util.List cn.bdqn.service.UserServiceImpl.queryAll())
1.6.2、返回值可以使用通配符,表示任意返回值
* cn.bdqn.service.UserServiceImpl.queryAll())
1.6.3、包名可以使用通配符表示任意包。有幾級包,就幾個*
* *.*.*.UserServiceImpl.queryAll())
但是對于包來說,連續(xù)的寫3個*,顯然也是麻煩的,那么可以使用“…”表示當前包及其子包。
// 表示的是任意包下的只要有UserServiceImpl類都會對queryAll方法進行增強
* *..UserServiceImpl.queryAll())
1.6.4、類名也可以用*
* *..*.queryAll()
1.6.5、方法也可以用*
* *..*.*()
1.6.6、參數列表
寫法1、可以直接寫數據類型:
基本類型直接寫名稱
int、double
引用類型寫包名.類名的方式
java.lang.String、java.util.List
寫法2、可以使用通配符表示任意類型
前提是必須要有參數。
寫法3、使用..
可以使用..表示有無參數均可,如果有參數則表示的可以是任意類型
1.6.7、全通配符寫法
* *..*.*(..)
1.6.8、使用最多的寫法
? 實際中的寫法:切到業(yè)務層實現類下的所有方法。即:
* 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è)務方法執(zhí)行完了,日志打印");
}
// 該方法的作用是在切入點方法執(zhí)行出錯后執(zhí)行
public void afterThrowingPrintLog(){
System.out.println("異常通知(afterThrowingPrintLog):業(yè)務方法出現異常了,日志打印");
}
// 該方法的作用是在切入點方法執(zhí)行之后不管有沒有錯誤,都最終要執(zhí)行
public void afterPrintLog(){
System.out.println("最終通知(afterPrintLog):業(yè)務方法不管有沒有異常了,日志打印");
}
}
1.7.2、配置AOP
<beans>
<!-- 2.1 把通知/增強Bean也需要注冊到Spring容器中 -->
<bean id="logger" class="cn.bdqn.advice.Logger"/>
<!-- 2.2 使用此標簽來去聲明開始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í)行產生異常之后執(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):開始打印日志啦
查詢全部用戶執(zhí)行啦
后置通知(afterReturningPrintLog):業(yè)務方法執(zhí)行完了,日志打印
最終通知(afterPrintLog):業(yè)務方法不管有沒有異常了,日志打印
**/
1.8、切入點表達式改進
? 通過11.7可以發(fā)現,我們在配置文件中配置了四種通知類型,其中的pointcut配置的是切入點表達式,發(fā)現是一模一樣的,那么有沒有一種改進寫法呢?可以將表達式抽取出來,將來可以引用。
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 使用此標簽來去聲明開始AOP的配置了-->
<aop:config>
<!--配置切面 -->
<aop:aspect id="loggerAdvice" ref="logger">
<!--
配置切入點表達式
id屬性用于指定切入點表達式的唯一標識。
expression屬性用于指定表達式內容
此標簽寫在aop:aspect標簽內部只能當前切面使用。
-->
<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í)行產生異常之后執(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標簽寫在了aop:aspect里面,這樣的話這切入點表達式只能被當前的切面使用,而如果其他切面想使用就使用不到了,所以我們可以把這個切入點表示再定義到外面。
<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 使用此標簽來去聲明開始AOP的配置了-->
<aop:config>
<!--
配置切入點表達式
id屬性用于指定切入點表達式的唯一標識。
expression屬性用于指定表達式內容
此標簽寫在aop:aspect標簽外面,那么所有的切面都可以使用。
-->
<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í)行產生異常之后執(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 使用此標簽來去聲明開始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ā)現:僅僅打印了環(huán)繞通知的代碼。當我們配置了環(huán)繞通知之后,切入點方法沒有執(zhí)行,而通知方法執(zhí)行了
*/
1.9.4、解決
? Spring框架為我們提供了一個接口:ProceedingJoinPoint。該接口有一個方法proceed(),此方法就相當于明確調用切入點方法。該接口可以作為環(huán)繞通知的方法參數,在程序執(zhí)行時,spring框架會為我們提供該接口的實現類供我們使用。
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 Boot 3.x》
作者簡介
楊開振——長期從事Java開發(fā)工作,擁有近十年的Java開發(fā)經驗,目前就職于一家互聯(lián)網金融公司,擔任互聯(lián)網軟件開發(fā)職位。
IT技術的狂熱愛好者,熱衷于Java互聯(lián)網方向的軟件技術開發(fā)與研究。熟練掌握Java基礎、軟件開發(fā)設計模式和數據庫相關知識,對Spring、MyBatis等主流Java開源框架有深入研究。
購書鏈接:點此進入文章來源:http://www.zghlxwxcb.cn/news/detail-854980.html
文章來源地址http://www.zghlxwxcb.cn/news/detail-854980.html
到了這里,關于【Spring進階系列丨第九篇】基于XML的面向切面編程(AOP)詳解的文章就介紹完了。如果您還想了解更多內容,請在右上角搜索TOY模板網以前的文章或繼續(xù)瀏覽下面的相關文章,希望大家以后多多支持TOY模板網!