一、AOP概述
AOP(Aspect Orient Programming),面向切面编程,是面向对象编程OOP的一种补充。面向对象编程是从静态角度考虑程序的结构,而面向切面编程是从动态角度考虑程序运行过程。
AOP底层就是采用动态代理模式实现的,采用了两种代理:JDK的动态代理与CGLIB的动态代理。
面向切面编程,就是将交叉业务逻辑封装成切面,利用AOP容器的功能将切面织入到主业务逻辑中。所谓交叉业务逻辑是指,通用的、与主业务逻辑无关的代码。如安全检查、事务、日志等。
若不是用AOP,则会出现代码纠缠,即交叉业务逻辑与主业务逻辑混合在一起,这样会使主业务逻辑变的混杂不清。
二、通知Advice
1、通知详解
(1)前置通知MethodBeforeAdvice
定义前置通知,需要实现MethodBeforeAdvice接口。该接口中有一个方法before(),会在目标方法执行之前执行。
前置通知的特点:
1、在目标方法执行之前执行。
2、不改变目标方法的执行流程,前置通知代码不能阻止目标方法执行。
3、不改变目标方法执行的结果。
举例:
创建IService接口:
package com.ietree.spring.basic.aop.beforeadvice; public interface IService { void doFirst(); void doSecond(); }
创建接口的实现类:
package com.ietree.spring.basic.aop.beforeadvice; public class SomeServiceImpl implements IService { @Override public void doFirst() { System.out.println("执行doFirst()方法"); } @Override public void doSecond() { System.out.println("执行doFirst()方法"); } }
创建前置通知类MyMethodBeforeAdvice,该类必须实现MethodBeforeAdvice接口:
package com.ietree.spring.basic.aop.beforeadvice; import java.lang.reflect.Method; import org.springframework.aop.MethodBeforeAdvice; /** * 前置通知 * * @author Root */ public class MyMethodBeforeAdvice implements MethodBeforeAdvice { // 当前方法在目标方法执行之前执行 // method:目标方法 // args:目标方法参数列表 // target:目标对象 @Override public void before(Method method, Object[] args, Object target) throws Throwable { // 对于目标方法的增强代码就写在这里 System.out.println("执行前置通知..."); } }
配置XML文件:
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation=" http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> <!-- 1、注册目标对象 --> <bean id="someService" class="com.ietree.spring.basic.aop.beforeadvice.SomeServiceImpl"/> <!-- 2、注册切面:通知 --> <bean id="myAdvice" class="com.ietree.spring.basic.aop.beforeadvice.MyMethodBeforeAdvice"/> <!-- 3、生成代理对象 --> <bean id="serviceProxy" class="org.springframework.aop.framework.ProxyFactoryBean"> <!-- 指定目标对象 --> <!-- <property name="targetName" value="someService"/> --> <property name="target" ref="someService"/> <!-- 指定切面 --> <property name="interceptorNames" value="myAdvice"/> </bean> </beans>
测试:
package com.ietree.spring.basic.aop.beforeadvice; import org.junit.Test; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; public class MyTest { @Test public void test() { String resource = "com/ietree/spring/basic/aop/beforeadvice/applicationContext.xml"; ApplicationContext ac = new ClassPathXmlApplicationContext(resource); IService service = (IService)ac.getBean("serviceProxy"); service.doFirst(); System.out.println("=============="); service.doSecond(); } }
输出:
执行前置通知... 执行doFirst()方法 ============== 执行前置通知... 执行doFirst()方法
注意:执行之前需要导入spring-aop-4.3.9.RELEASE.jar包
(2)后置通知AfterReturningAdvice
定义前置通知,需要实现AfterReturningAdvice接口。该接口中有一个方法afterReturning(),会在目标方法执行之后执行。
后置通知的特点:
1、在目标方法执行之后执行。
2、不改变目标方法的执行流程,后置通知代码不能阻止目标方法执行。
3、不改变目标方法执行的结果。
大致流程和前置通知差不多,这里就简单列举一下不同之处:
创建后置通知类并实现AfterReturningAdvice接口,重写afterReturning()方法:
package com.ietree.spring.basic.aop.afterreturningadvice; import java.lang.reflect.Method; import org.springframework.aop.AfterReturningAdvice; /** * 后置通知:可以获取到目标方法的返回结果,但是无法改变目标方法的结果 * * @author Root */ public class MyAfterReturningAdvice implements AfterReturningAdvice { // 在目标方法执行之后执行 // returnValue:目标方法的返回值 @Override public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable { System.out.println("执行后置通知方法 returnValue = " + 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" xsi:schemaLocation=" http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> <!-- 1、注册目标对象 --> <bean id="someService" class="com.ietree.spring.basic.aop.afterreturningadvice.SomeServiceImpl"/> <!-- 2、注册切面:通知 --> <bean id="myAdvice" class="com.ietree.spring.basic.aop.afterreturningadvice.MyAfterReturningAdvice"/> <!-- 3、生成代理对象 --> <bean id="serviceProxy" class="org.springframework.aop.framework.ProxyFactoryBean"> <!-- 指定目标对象 --> <property name="target" ref="someService"/> <!-- 指定切面 --> <property name="interceptorNames" value="myAdvice"/> </bean> </beans>
注意:后置通知可以获取到目标方法的返回结果,但是无法改变目标方法执行的返回结果。
(3)环绕通知MethodInterceptor
定义环绕通知,需要实现MethodInterceptor接口。环绕通知,也叫方法拦截器,可以在目标方法调用之前及之后做处理,可以改变目标方法的返回值,也可以改变程序执行流程。
创建环绕通知类MyMethodInterceptor,实现MethodInterceptor接口:
package com.ietree.spring.basic.aop.methodinterceptor; import org.aopalliance.intercept.MethodInterceptor; import org.aopalliance.intercept.MethodInvocation; // 环绕通知:可以修改目标方法的返回结果 public class MyMethodInterceptor implements MethodInterceptor { @Override public Object invoke(MethodInvocation invocation) throws Throwable { System.out.println("执行环绕通知:目标方法执行之前"); // 执行目标方法 Object result = invocation.proceed(); System.out.println("执行环绕通知:目标方法执行之后"); if(null != result) { result = ((String)result).toUpperCase(); } return result; } }
配置文件:
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation=" http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> <!-- 1、注册目标对象 --> <bean id="someService" class="com.ietree.spring.basic.aop.methodinterceptor.SomeServiceImpl"/> <!-- 2、注册切面:通知 --> <bean id="myAdvice" class="com.ietree.spring.basic.aop.methodinterceptor.MyMethodInterceptor"/> <!-- 3、生成代理对象 --> <bean id="serviceProxy" class="org.springframework.aop.framework.ProxyFactoryBean"> <!-- 指定目标对象 --> <!-- <property name="targetName" value="someService"/> --> <property name="target" ref="someService"/> <!-- 指定切面 --> <property name="interceptorNames" value="myAdvice"/> </bean> </beans>
测试:
package com.ietree.spring.basic.aop.methodinterceptor; import org.junit.Test; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; public class MyTest { @Test public void test() { String resource = "com/ietree/spring/basic/aop/methodinterceptor/applicationContext.xml"; ApplicationContext ac = new ClassPathXmlApplicationContext(resource); IService service = (IService)ac.getBean("serviceProxy"); service.doFirst(); System.out.println("=============="); String result = service.doSecond(); System.out.println(result); } }
输出:
执行环绕通知:目标方法执行之前 执行doFirst()方法 执行环绕通知:目标方法执行之后 ============== 执行环绕通知:目标方法执行之前 执行doSecond()方法 执行环绕通知:目标方法执行之后 ABCDE
注意:环绕通知不仅可以获取到方法的返回结果,而且还可以修改方法的返回结果。
(4)异常通知ThrowsAdvice
异常分两种:
1)运行时异常,不进行处理,也可以通过编译。若一个类继承自RunTimeException,则该异常就是运行时异常
2)编译时异常,受查异常,Checked Exception。不进行处理,则无法通过编译。若一个类继承自Exception,则该异常就是受查异常。
创建接口IService:
package com.ietree.spring.basic.aop.throwsadvice; public interface IService { // 用户登录 boolean login(String username, String password) throws UserException; }
创建接口实现类SomeServiceImpl:
package com.ietree.spring.basic.aop.throwsadvice; public class SomeServiceImpl implements IService { @Override public boolean login(String username, String password) throws UserException { if (!"jack".equals(username)) { throw new UsernameException("用户名错误!"); } if (!"123".equals(password)) { throw new PasswordException("密码错误!"); } // double i = 3 / 0; return true; } }
创建三个自定义异常:
package com.ietree.spring.basic.aop.throwsadvice; /** * 自定义异常 * 异常分两种: * 1)运行时异常,不进行处理,也可以通过编译。若一个类继承自RunTimeException,则该异常就是运行时异常 * 2)编译时异常,受查异常,Checked Exception。不进行处理,则无法通过编译。若一个类继承自Exception,则该异常就是受查异常。 * * @author Root */ public class UserException extends Exception { public UserException() { super(); } public UserException(String message) { super(message); } }
用户名异常:
package com.ietree.spring.basic.aop.throwsadvice; public class UsernameException extends UserException { public UsernameException() { super(); } public UsernameException(String message) { super(message); } }
密码异常:
package com.ietree.spring.basic.aop.throwsadvice; public class PasswordException extends UserException { public PasswordException() { super(); } public PasswordException(String message) { super(message); } }
定义异常通知:
package com.ietree.spring.basic.aop.throwsadvice; import org.springframework.aop.ThrowsAdvice; /** * 异常通知 当目标方法抛出与指定类型的异常具有is-a关系的异常时,执行当前方法afterThrowing() * * @author Root */ public class MyThrowsAdvice implements ThrowsAdvice { // 当目标方法抛出UsernameException异常时,执行当前方法 public void afterThrowing(UsernameException ex) { System.out.println("发生用户名异常 ex = " + ex.getMessage()); } // 当目标方法抛出UsernameException异常时,执行当前方法 public void afterThrowing(PasswordException ex) { System.out.println("发生密码异常 ex = " + ex.getMessage()); } // 当目标方法抛出UsernameException异常时,执行当前方法 public void afterThrowing(Exception ex) { System.out.println("发生其它异常 ex = " + ex.getMessage()); } }
配置文件:
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation=" http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> <!-- 1、注册目标对象 --> <bean id="someService" class="com.ietree.spring.basic.aop.throwsadvice.SomeServiceImpl"/> <!-- 2、注册切面:通知 --> <bean id="myAdvice" class="com.ietree.spring.basic.aop.throwsadvice.MyThrowsAdvice"/> <!-- 3、生成代理对象 --> <bean id="serviceProxy" class="org.springframework.aop.framework.ProxyFactoryBean"> <!-- 指定目标对象 --> <!-- <property name="targetName" value="someService"/> --> <property name="target" ref="someService"/> <!-- 指定切面 --> <property name="interceptorNames" value="myAdvice"/> </bean> </beans>
测试:
package com.ietree.spring.basic.aop.throwsadvice; import org.junit.Test; import org.springframework.context.ApplicationContext; import org.springframework.context.support.ClassPathXmlApplicationContext; public class MyTest { @Test public void test() throws UserException { String resource = "com/ietree/spring/basic/aop/throwsadvice/applicationContext.xml"; ApplicationContext ac = new ClassPathXmlApplicationContext(resource); IService service = (IService)ac.getBean("serviceProxy"); service.login("jack", "123"); } }
(5)同时使用多个通知的配置方法
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation=" http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd"> <!-- 1、注册目标对象 --> <bean id="someService" class="com.ietree.spring.basic.aop.multipleadvice.SomeServiceImpl"/> <!-- 2、注册切面:通知 --> <!-- 前置通知 --> <bean id="beforeAdvice" class="com.ietree.spring.basic.aop.multipleadvice.MyMethodBeforeAdvice"/> <!-- 后置通知 --> <bean id="afterAdvice" class="com.ietree.spring.basic.aop.multipleadvice.MyAfterReturningAdvice"/> <!-- 3、生成代理对象 --> <bean id="serviceProxy" class="org.springframework.aop.framework.ProxyFactoryBean"> <!-- 指定目标对象 --> <property name="target" ref="someService"/> <!-- 指定切面 --> <!-- 方式一 --> <property name="interceptorNames" value="beforeAdvice,afterAdvice"/> <!-- 方式二 --> <!-- <property name="interceptorNames"> <array> <value>beforeAdvice</value> <value>afterAdvice</value> </array> </property> --> </bean> </beans>
三、顾问Advisor
四、自动代理生成器
五、AspectJ对AOP的实现