什么是AOP
AOP(Aspect Oriented Programming),面向切面思想,是Spring的三大核心思想之一(另外两个:IOC-控制反转、DI-依赖注入)。
AOP体系
简单地去理解,其实AOP要做三类事:
- 在哪里切入,也就是权限校验等非业务操作在哪些业务代码中执行。
- 在什么时候切入,是业务代码执行前还是执行后。
- 切入后做什么事,比如做权限校验、日志记录等。
SpringAOP概念
- Pointcut:切点,决定处理如权限校验、日志记录等在何处切入业务代码中(即织入切面)。切点分为execution方式和annotation方式。前者可以用路径表达式指定哪些类织入切面,后者可以指定被哪些注解修饰的代码织入切面。
- Advice:处理,包括处理时机和处理内容。处理内容就是要做什么事,比如校验权限和记录日志。处理时机就是在什么时机执行处理内容,分为前置处理(即业务代码执行前)、后置处理(业务代码执行后)等。
- Aspect:切面,即Pointcut和Advice。
- Joint point:连接点,是程序执行的一个点。例如,一个方法的执行或者一个异常的处理。在 Spring AOP 中,一个连接点总是代表一个方法执行。
- Weaving:织入,就是通过动态代理,在目标对象方法中执行处理内容的过程。
案例
添加Maven依赖
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
实例
- 自定义一个注解LogAnnotation
- 创建一个切面类,切点设置为拦截所有标注LogAnnotation的方法,截取到接口的参数,实现日志记录
- 将LogAnnotation标注在测试接口类的测试接口test上
LogAnnotation
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface LogAnnotation{
}
Aspect
@Component
@Aspect
@Slf4j
public class LogAspect {
@Pointcut("@annotation(com.example.quartz.annotation.LogAnnotation)")
private void logAdvicePointCut(){}
@Around("logAdvicePointCut()")
public Object logAdvice(ProceedingJoinPoint joinPoint) throws Throwable{
long startTime = System.currentTimeMillis();
Object obj = null;
try {
obj = joinPoint.proceed();
return obj;
}
finally {
long costTime = System.currentTimeMillis() - startTime;
MethodSignature signature = (MethodSignature)joinPoint.getSignature();
log.info("method={}, args={}, cost_time={}, result={}", signature.getName(), signature.getParameterNames(), costTime, obj);
}
}
}
在Controller的方法上加上LogAnnotation注解
AOP结果