1、AOP (面向切面编程)
AOP为Aspect Oriented Programming的缩写,意为:面向切面编程
通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术
AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型
利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率
AOP 思想:横向重复、纵向抽取
2、底层实现: Spring 的 AOP 的底层用到两种代理机制:
JDK 的动态代理:针对实现了接口的类产生代理
Cglib 的动态代理:针对没有实现接口的类产生代理,应用的是底层的字节码增强的技术 生成当前类的子类对象
3、AOP 的开发中的相关术语:
Joinpoint(连接点):所谓连接点是指那些被拦截到的点。在 spring 中,这些点指的是方法,因为 spring 只支持方法类型的连接点(可以被切入的点)
Pointcut(切入点):所谓切入点是指我们要对哪些 Joinpoint 进行拦截的定义。(已经被切入的点)
Advice(通知/增强):所谓通知是指拦截到 Joinpoint 之后所要做的事情就是通知。通知分为前置通知,后置通知,异常通知,最终通知,环绕通知(切面要完成的功能)
Introduction(引介):引介是一种特殊的通知在不修改类代码的前提下,Introduction 可以在运行期为类动态地添加一些方法或 Field。
Aspect(切面):是切入点和通知(引介)的结合
Target(目标对象):代理的目标对象
Proxy(代理):一个类被 AOP 织入增强后,就产生一个结果代理类。
Weaving(织入):是指把增强应用到目标对象来创建新的代理对象的过程。spring 采用动态代理织入,而 AspectJ 采用编译期织入和类装载期织入。
4、 示例的项目结构:
4.1使用XML配置
<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-2.5.xsd"> <!-- 目标对象 --> <bean name="userService" class="com.roxy.spring.service.UserServiceImpl"></bean> <!-- 通知对象 --> <bean name="transactionAdvice" class="com.roxy.spring.advice.TransactionAdvice"></bean> <!-- 将通知对象织入目标对象 --> <aop:config> <!-- 选择切入点 --> <aop:pointcut expression="execution(* com.roxy.spring.service..*ServiceImpl.*(..))" id="pointcut"/> <!-- 选择切入时机 --> <aop:aspect ref="transactionAdvice"> <aop:before method="before" pointcut-ref="pointcut"/> <aop:after-returning method="afterReturning" pointcut-ref="pointcut"/> <aop:after method="after" pointcut-ref="pointcut"/><!-- 无论是否出现异常都会调用 --> <!--<aop:around method="around" pointcut-ref="pointcut"/> <aop:after-throwing method="afterException" pointcut-ref="pointcut"/> </aop:aspect> </aop:config> </beans>
public interface UserService { public void save(); public void delete(); public void update(); public void select(); }
public class UserServiceImpl implements UserService{ @Override public void save() { System.out.println("保存用户"); } @Override public void delete() { System.out.println("删除用户"); } @Override public void update() { System.out.println("更新用户"); } @Override public void select() { System.out.println("查询用户"); } }
public class TransactionAdvice { public void before() { System.out.println("前置通知被执行"); } public void afterReturning() { System.out.println("后置通知被执行(出现异常不调用)"); } public void after() { System.out.println("后置通知被执行(无论是否出现异常都会调用)"); } public void afterException() { System.out.println("异常通知被执行"); } public Object around(ProceedingJoinPoint point) throws Throwable { System.out.println("around_before"); Object proceed = point.proceed();//调用目标方法 System.out.println("around_after"); return proceed; } }
4.2 使用注解
<!-- 目标对象 --> <bean name="userService" class="com.roxy.spring.service.UserServiceImpl"></bean> <!-- 通知对象 --> <bean name="transactionAdvice" class="com.roxy.spring.advice.TransactionAdvice"></bean> <!-- 开启织入注解 --> <aop:aspectj-autoproxy></aop:aspectj-autoproxy>
@Aspect //该类为切面 public class TransactionAdvice { @Pointcut(value="execution(* com.roxy.spring.service..*ServiceImpl.*(..))") public void PointCut(){} @Before("TransactionAdvice.PointCut()") public void before() { System.out.println("前置通知被执行"); } @AfterReturning("TransactionAdvice.PointCut()") public void afterReturning() { System.out.println("后置通知被执行(出现异常不调用)"); } @After("TransactionAdvice.PointCut()") public void after() { System.out.println("后置通知被执行(无论是否出现异常都会调用)"); } @AfterThrowing("TransactionAdvice.PointCut()") public void afterException() { System.out.println("异常通知被执行"); } @Around("TransactionAdvice.PointCut()") public Object around(ProceedingJoinPoint point) throws Throwable { System.out.println("around_before"); Object proceed = point.proceed();//调用目标方法 System.out.println("around_after"); return proceed; } }
5、5种通知的调用顺序
console:
around_before
前置通知被执行
保存用户
around_after
后置通知被执行(无论是否出现异常都会调用)
后置通知被执行(出现异常不调用)
result:
环绕通知 --> 前置通知 --> 调用方法 --> 环绕通知 --> 后置通知(after) --> 后置通知(afterException)
6、AOP中的正则表达式
aop配置切点的正则表达式 . 匹配除换行外的所有单个字符 * 匹配前面的字符0次或多次 + 匹配前面的字符1次或多次 ? 匹配前面的字符0次或1次 ^ 匹配字符串开始 $ 匹配字符串结束 x|y 匹配x或y [xyz] zyx其中一个 {n} 匹配n次 {n,} 匹配至少n次 {n,m} 匹配n次到m次 \ 转意字符