前言
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】文章目录