常见Spring内置AOP接口
Before通知
- 在目标方法被调用前调用,
- 切面需要实现的接口:org.springframework.aop.MethodBeforeAdvice
After通知
- 在目标方法被调用后调用
- 切面需要实现的接口:org.springframework.aop.AfterReturningAdvice
Throws通知
- 在目标方法抛出异常时调用
- 切面需要实现的接口:org.springframework.aop.ThrowsAdvice
Around通知
- 环绕通知:拦截对目标对象方法的调用,在被调用方法前后执行切面功能,例如:事务切面就是环绕通知
- 切面需要实现的接口:org.aopalliance.intercept.MethodInterceptor
原生Before通知示例
- 其他通知类型同理,不再重复演示
业务接口
package com.example.service; /** * 定义业务接口 */ public interface Service { //购买功能 default void buy(){} //预定功能 default String order(int orderNums){return null;} }
业务实现类
package com.example.service.impl; import com.example.service.Service; /** * 图书业务实现类 */ public class BookServiceImpl implements Service { @Override public void buy() { System.out.println("图书购买业务...."); } @Override public String order(int orderNums) { System.out.println("预定图书: " + orderNums + " 册"); return "预定成功"; } }
切面实现类
- 相当于使用了Spring内置的AOP前置通知接口:org.springframework.aop.MethodBeforeAdvice
package com.example.advice; import org.springframework.aop.MethodBeforeAdvice; import java.lang.reflect.Method; import java.text.SimpleDateFormat; import java.util.Arrays; import java.util.Date; public class LogAdvice implements MethodBeforeAdvice { @Override public void before(Method method, Object[] args, Object target) throws Throwable { //3个参数:目标方法,目标方法返回值,目标对象 SimpleDateFormat sf = new SimpleDateFormat("yyyy-MM-dd"); System.out.println("[业务功能名称] :" + method.getName()); System.out.println("[业务参数信息] :" + Arrays.toString(args)); System.out.println("[业务办理时间] :" + sf.format(new Date())); System.out.println("--------- 具体业务如下 ---------"); } }
业务功能和切面功能整合
- 使用applicationContext.xml,看起来更加直观
<!--创建业务对象--> <bean id="bookServiceTarget" class="com.example.service.impl.BookServiceImpl"/> <!--创建切面的对象--> <bean id="logAdvice" class="com.example.advice.LogAdvice"/> <!-- 相当于创建动态代理对象,用来在底层绑定业务和切面--> <bean id="bookService" class="org.springframework.aop.framework.ProxyFactoryBean"> <!--配置业务接口,底层的jdk动态代理需要用--> <property name="interfaces" value="com.example.service.Service"/> <!--配置切面,可以有多个--> <property name="interceptorNames"> <list> <value>logAdvice</value> </list> </property> <!--待织入切面的业务功能对象,底层的jdk动态代理需要用--> <property name="target" ref="bookServiceTarget"/> </bean>
对比手写的AOP版本5
将上述applicationContext.xml的内容和AOP版本5中的ProxyFactory对比,上述xml作用就相当于我们手写的ProxyFactory作用:
获取到业务功能对象和切面功能对象,并将他们传给底层来获取动态代理对象,在底层完成切面功能的织入
不管是xml或者通过注解来整合业务和切面,Spring底层都是像手写的AOP版本5一样,通过jdk动态代理来实现的,只不过现在封装起来了
- AOP版本5中的ProxyFactory代理工厂
package com.example.proxy05; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; import java.lang.reflect.Proxy; /** * 代理工厂,获取动态代理对象 */ public class ProxyFactory { //获取jdk动态代理对象 //Service target 接口类型的业务功能对象 //Aop aop 接口功能的切面功能对象 public static Object getProxy(Service target, Aop aop){ //使用内置类,返回jdk动态代理对象 return Proxy.newProxyInstance( target.getClass().getClassLoader(), //获取实现的所有接口 target.getClass().getInterfaces(), //调用目标对象的目标方法 new InvocationHandler() { @Override public Object invoke( Object obj, Method method, Object[] args) throws Throwable { //存放目标对象的目标方法的返回值 Object res = null; try{ //切面功能 aop.before(); //业务功能,根据外部调用的功能,动态代理目标对象被调用的方法 res = method.invoke(target, args); //切面功能 aop.after(); }catch (Exception e){ //切面功能 aop.exception(); } return res; } } ); } }
测试
package test; import com.example.service.Service; import org.junit.Test; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; public class TestSpringAOP { @Test public void testSpringAop(){ //创建Spring容器 ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml"); //获取动态代理对象 Service agent = (Service) ac.getBean("bookService"); //调用业务功能 String res = agent.order(10); System.out.println("结果: " + res); } }
测试输出
- Spring原生AOP前置通知在业务功能前顺利执行。底层的jdk动态代理也正确的反射调用了外部调用的目标方法,正确接收了参数并给出了返回值
[业务功能名称] :order [业务参数信息] :[10] [业务办理时间] :2022-08-23 --------- 具体业务如下 --------- 预定图书: 10 册 结果: 预定成功 Process finished with exit code 0
注意
- 在开发中一般不常用Spring原生的AOP支持,在需要时,常使用其他专门的AOP框架
- 手写AOP框架和简单演示Spring内置AOP通知接口是为了更好的理解AOP面向接口编程的思想