前言
Spring AOP 的實現有三種方式:使用Spring API 接口、自定義切面、使用註解。這裏將第一種,使用Spring API 接口,另外另種在我的接下來兩篇文章中。
一、使用Spring API接口
使用原生Spring API接口實現數據訪問層前後增加日誌。
注意:使用AOP前,除了spring以外,還需要在maven導入增加一個aspectjweaver的依賴包。
<!-- aspectjweaver -->
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.9.4</version>
</dependency>
項目結構:
UserDao接口:
/** 用戶數據訪問層接口 */
public interface UserDao {
void ins();
void del();
void upd();
void sel();
}
UserDao接口實現類:
public class UserDaoImpl implements UserDao {
@Override
public void ins() {
System.out.println("insert命令");
}
@Override
public void del() {
System.out.println("delete命令");
}
@Override
public void upd() {
System.out.println("update命令");
}
@Override
public void sel() {
System.out.println("select命令");
}
}
前置通知:
/** 前置通知(方法前加通知) */
public class LogBefore implements MethodBeforeAdvice {
/**
* 參數:method:要執行的目標對象方法。objects:參數(args)。o:目標對象(target)
*/
@Override
public void before(Method method, Object[] args, Object target) throws Throwable {
System.out.println("----- 執行了 " + target.getClass().getName() + " 類的 " + method.getName() + " 方法 -----");
}
}
後置通知:
/** 後置通知(方法之後加通知) */
public class LogAfter implements AfterReturningAdvice {
@Override
public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable {
System.out.println("----- 執行了 " + target.getClass().getName() + " 類的 " + method.getName() + " 方法,返回值類型爲:"+returnValue+" -----");
}
}
配置文件:
<?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: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/aop
http://www.springframework.org/schema/aop/spring-aop.xsd">
<!--註冊bean-->
<bean id="userDaoImpl" class="com.shengjava.aop.dao.impl.UserDaoImpl"/>
<bean id="logAfter" class="com.shengjava.aop.log.LogAfter"/>
<bean id="logBefore" class="com.shengjava.aop.log.LogBefore"/>
<!--方式一:使用Spring API接口-->
<!--注意:需要導入aop的約束-->
<aop:config>
<!--切入點:expression:表達式,execution(要執行的位置!) *(..):* 表示所有的方法 (..)表示任何參數-->
<aop:pointcut id="pc-userDao" expression="execution(* com.shengjava.aop.dao.impl.UserDaoImpl.*(..))"/>
<!--前置通知-->
<aop:advisor advice-ref="logBefore" pointcut-ref="pc-userDao"/>
<!--後置通知-->
<aop:advisor advice-ref="logAfter" pointcut-ref="pc-userDao"/>
</aop:config>
</beans>
測試:
public class Test {
public static void main(String[] args) {
ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
// 動態代理(代理的是接口,所以類型應該也是接口類型)
UserDao ud = context.getBean("userDaoImpl", UserDao.class);
// 執行
ud.ins();
}
}
測試結果:使用了Spring的aop,各個類職責非常清晰。
----- 執行了 com.shengjava.aop.dao.impl.UserDaoImpl 類的 ins 方法 -----
insert命令
----- 執行了 com.shengjava.aop.dao.impl.UserDaoImpl 類的 ins 方法,返回值類型爲:null -----
擴展:配置文件中的切入點expression表達式:Combining Pointcut Expressions
相關
我的該分類的其他相關文章,請點擊:【Spring + Spring MVC + MyBatis】文章目錄