Aop
<!-- 配置srping的Ioc,把service對象配置進來-->
<bean id="accountService" class="com.itheima.service.impl.AccountServiceImpl"></bean>
<!--spring中基於XML的AOP配置步驟
1、把通知Bean也交給spring來管理
2、使用aop:config標籤表明開始AOP的配置
3、使用aop:aspect標籤表明配置切面
id屬性:是給切面提供一個唯一標識
ref屬性:是指定通知類bean的Id。
4、在aop:aspect標籤的內部使用對應標籤來配置通知的類型
我們現在示例是讓printLog方法在切入點方法執行之前之前:所以是前置通知
aop:before:表示配置前置通知
method屬性:用於指定Logger類中哪個方法是前置通知
pointcut屬性:用於指定切入點表達式,該表達式的含義指的是對業務層中哪些方法增強
切入點表達式的寫法:
關鍵字:execution(表達式)
表達式:
訪問修飾符 返回值 包名.包名.包名...類名.方法名(參數列表)
標準的表達式寫法:
public void com.itheima.service.impl.AccountServiceImpl.saveAccount()
訪問修飾符可以省略
void com.itheima.service.impl.AccountServiceImpl.saveAccount()
返回值可以使用通配符,表示任意返回值
* com.itheima.service.impl.AccountServiceImpl.saveAccount()
包名可以使用通配符,表示任意包。但是有幾級包,就需要寫幾個*.
* *.*.*.*.AccountServiceImpl.saveAccount())
包名可以使用..表示當前包及其子包
* *..AccountServiceImpl.saveAccount()
類名和方法名都可以使用*來實現通配
* *..*.*()
參數列表:
可以直接寫數據類型:
基本類型直接寫名稱 int
引用類型寫包名.類名的方式 java.lang.String
可以使用通配符表示任意類型,但是必須有參數
可以使用..表示有無參數均可,有參數可以是任意類型
全通配寫法:
* *..*.*(..)
實際開發中切入點表達式的通常寫法:
切到業務層實現類下的所有方法
* com.itheima.service.impl.*.*(..)
-->
<!-- 配置Logger類 -->
<bean id="logger" class="com.itheima.utils.Logger"></bean>
<!--配置AOP-->
<aop:config>
<!--配置切面 -->
<aop:aspect id="logAdvice" ref="logger">
<!-- 配置通知的類型,並且建立通知方法和切入點方法的關聯-->
<aop:before method="printLog" pointcut="execution(* com.itheima.service.impl.*.*(..))"></aop:before>
</aop:aspect>
</aop:config>
四種常用通知
pointchut表達式通用化
標籤 和pointcut-ref屬性
也可以挪到aspect外面,其他切面也可以用(寫在切面之前,需要遵守約束)
環繞通知
aop 註解方式
package com.itheima.utils;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;
/**
* 用於記錄日誌的工具類,它裏面提供了公共的代碼
*/
@Component("logger")
@Aspect//表示當前類是一個切面類
public class Logger {
@Pointcut("execution(* com.itheima.service.impl.*.*(..))")
private void pt1(){}
/**
* 前置通知
*/
// @Before("pt1()")
public void beforePrintLog(){
System.out.println("前置通知Logger類中的beforePrintLog方法開始記錄日誌了。。。");
}
/**
* 後置通知
*/
// @AfterReturning("pt1()")
public void afterReturningPrintLog(){
System.out.println("後置通知Logger類中的afterReturningPrintLog方法開始記錄日誌了。。。");
}
/**
* 異常通知
*/
// @AfterThrowing("pt1()")
public void afterThrowingPrintLog(){
System.out.println("異常通知Logger類中的afterThrowingPrintLog方法開始記錄日誌了。。。");
}
/**
* 最終通知
*/
// @After("pt1()")
public void afterPrintLog(){
System.out.println("最終通知Logger類中的afterPrintLog方法開始記錄日誌了。。。");
}
/**
* 環繞通知
* 問題:
* 當我們配置了環繞通知之後,切入點方法沒有執行,而通知方法執行了。
* 分析:
* 通過對比動態代理中的環繞通知代碼,發現動態代理的環繞通知有明確的切入點方法調用,而我們的代碼中沒有。
* 解決:
* Spring框架爲我們提供了一個接口:ProceedingJoinPoint。該接口有一個方法proceed(),此方法就相當於明確調用切入點方法。
* 該接口可以作爲環繞通知的方法參數,在程序執行時,spring框架會爲我們提供該接口的實現類供我們使用。
*
* spring中的環繞通知:
* 它是spring框架爲我們提供的一種可以在代碼中手動控制增強方法何時執行的方式。
*/
@Around("pt1()")
public Object aroundPringLog(ProceedingJoinPoint pjp){
Object rtValue = null;
try{
Object[] args = pjp.getArgs();//得到方法執行所需的參數
System.out.println("Logger類中的aroundPringLog方法開始記錄日誌了。。。前置");
rtValue = pjp.proceed(args);//明確調用業務層方法(切入點方法)
System.out.println("Logger類中的aroundPringLog方法開始記錄日誌了。。。後置");
return rtValue;
}catch (Throwable t){
System.out.println("Logger類中的aroundPringLog方法開始記錄日誌了。。。異常");
throw new RuntimeException(t);
}finally {
System.out.println("Logger類中的aroundPringLog方法開始記錄日誌了。。。最終");
}
}
}
註解方式存在的問題: 方法的調用順序存在問題 最終通知會在 後置,異常通知之前執行