AOP
AOP:Aspect Oriented Programming (面向切面编程、面向方面编程),其实就是面向特定方法编程
依赖
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
@Component
@Aspect
@Slf4j
public class TimeAspect {
@Around("execution(* com.example.demo.service.*.*(..))")
public Object recordTime(ProceedingJoinPoint joinPoint) throws Throwable {
long begin = System.currentTimeMillis();
Object result = joinPoint.proceed();
long end = System.currentTimeMillis();
log.info(joinPoint.getSignature()+"用时:{}",end-begin);
return result;
}
}
AOP核心概念
AOP执行流程
通知类型
切入点表达式
@Pointcut("execution(* com.example.demo.service.*.*(..))")
private void pt(){}
@Around("pt()")
通知执行顺序
类名靠前先执行
也可以用@Order(数字)来控制,数字越小越先执行。
切入点表达式
- execution(…):根据方法的签名来匹配
- @annotation(…):根据注解匹配
切入点表达式-execution
通配符
切入点表达式书写建议
- 所有业务方法名在命名时尽量规范,方便切入点表达式快速匹配。如:查询类方法都是 find 开头,更新类方法都是 update开头。
- 描述切入点方法通常基于接口描述而不是直接描述实现类增强拓展性
- 在满足业务需要的前提下,尽量缩小切入点的匹配范围。如: 包名匹配尽量不使用…,使用*匹配单个包
切入点表达式-annotation
@Retention(RetentionPolicy.RUNTIME)//运行时
@Target(ElementType.METHOD)//方法上
public @interface MyLog {
}
@Pointcut("@annotation(com.example.demo.aop.MyLog)")
连接点
- 在Spring中用JoinPoint抽象了连接点,用它可以获得方法执行时的相关信息,如目标类名、方法名、方法参数等
- 对于 @Around 通知,获取连接点信息只能使用 ProceedingJoinPoint
- 对于其他四种通知,获取连接点信息只能使用 JoinPoint,它是 ProceedingJoinPoint 的父类型
//获取目标对象类名
String name = joinPoint.getTarget().getClass().getName();
log.info("1:{}",name);
//获取目标对象的方法名
String name1 = joinPoint.getSignature().getName();
log.info("2:{}",name1);
//获取目标对象运行时传入的参数
Object[] args = joinPoint.getArgs();
log.info("3:{}", Arrays.toString(args));
//放行目标方法的执行
Object result = joinPoint.proceed();
//获取目标方法运行时的返回值
切面 类获取网络请求jwt令牌
@Component
@Aspect
@Slf4j
public class TimeAspect {
@Autowired
HttpServletRequest request;
@Pointcut("@annotation(com.example.demo.aop.MyLog)")
private void pt(){}
@Around("pt()")
public Object recordTime(ProceedingJoinPoint joinPoint) throws Throwable {
String jwt = request.getHeader("token");
Claims claims = JwtUtils.parseJwt(jwt);
Object result = joinPoint.proceed();
return result;
}
}