一、概述
Aspect Oriented Programming(面向切面编程、面向方面编程),面向特定方法变成
动态代理是面向切面编程最主流的实现。SpringAOP和Spring框架的高级技术,在管理bean对象的过程中,主要通过底层的动态代理机制,对特定的方法进行编程。
1.1、示例
通过@Aspect把类注入,使用@Around注解指定方法在哪个地方执行
例:
package com.example.aop;
import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.stereotype.Component;
@Slf4j
@Component
@Aspect
public class TimeAspect {
@Around("execution(* com.example.service.*.*(..))") //切入点表达式
public Object recordTime(ProceedingJoinPoint joinPoint) throws Throwable {
//1.记录开始时间
long begin=System.currentTimeMillis();
//2.调用原始方法执行
Object proceed = joinPoint.proceed();
//3.记录结束时间,计算方法执行时间
long end = System.currentTimeMillis();
log.info(joinPoint.getSignature()+"方法执行时间,{}ms",end-begin);
return proceed;
}
}
二、核心概念
连接点:JoinPoint,可以被AOP控制的方法(暗含方法执行时的相关信息)
通知:Advice,指哪些重复的逻辑,也就是共性功能(最终体现为一个方法)
切入点:PoinCut,匹配连接点的条件,通知仅会在切入点方法执行时被应用
目标对象:Target,通知所有应用的对象
2.1通知对象
通知类型
通知顺序
当有多个切面的切入点都匹配到目标方法,目标方法运行时,多个通知方法都会被执行
1.不同切面类中,默认按照切面类的类名字母排序
* 目标方法前的通知方法,字母排名靠前的先执行
* 目标方法后的通知方法,字母排名靠前的后执行
2.用@Order(数字)加在切面类上来控制顺序
* 目标方法前的通知方法,数字小的先执行
* 目标方法后的通知方法,数字小的后执行
三、切入点表达式
切入点表达式:描述切入点方法的一种表达式
作用:主要用来决定项目中的哪些方法需要加入通知
常见形式:
1. execution(.....):根据方法的签名来匹配
2. @annotation(......):根据注解匹配
通配符
* :单个独立的任意符号,可以通配任意返回值、包名、类名、方法名,任意类型的一个参数,也可以通配包、类、方法名的一部分
execution(* com.*.service.*.update*(*))
.. :多个连续的任意符号,可以通配任意层级的包,或任意类型、任意个数的参数
execution(* com.itheima..DeptService.*(..))
根据业务需要,可以使用&&、||、!来组合比较复杂的切入点表达式
3.1execution
语法:
execution(访问修饰符? 返回值 包名.类名.?方法名(方法参数) throws 异常?)
其中带?的表示可以省略的部分
访问修饰符:可省略(比如:public protected)
包名.类名:可省略
throws 异常:可省略(注意是方法上声明抛出的异常,不是实际抛出的异常)
3.2@annotation
用于匹配标识有特定注解的方法
通过注解的方式指定
四、连接点
在Spring中用JoinPoint抽象类连接点,用它可以获得方法执行时的相关信息,如目标类名、方法名、方法参数等
对于@Around通知,获取连接点信息只能使用 ProuceedingJoinPoint
对于其他四种通知,获取连接点信息只能使用JoinPoint,它是ProceedingJoinPoint的父类型