Spring Boot使用AOP
一、为什么需要面向切面编程?
面向对象编程(OOP)的好处是显而易见的,缺点也同样明显。当需要为多个不具有继承关系的对象添加一个公共的方法的时候,例如日志记录、性能监控等,如果采用面向对象编程的方法,需要在每个对象里面都添加相同的方法,这样就产生了较大的重复工作量和大量的重复代码,不利于维护。面向切面编程(AOP)是面向对象编程的补充,简单来说就是统一处理某一“切面”的问题的编程思想。如果使用AOP的方式进行日志的记录和处理,所有的日志代码都集中于一处,不需要再每个方法里面都去添加,极大减少了重复代码。
二、Spring AOP术语
通知(Advice)包含了需要用于多个应用对象的横切行为,完全听不懂,没关系,通俗一点说就是定义了“什么时候”和“做什么”。
连接点(Join Point)是程序执行过程中能够应用通知的所有点。
切点(Poincut)是定义了在“什么地方”进行切入,哪些连接点会得到通知。显然,切点一定是连接点。
切面(Aspect)是通知和切点的结合。通知和切点共同定义了切面的全部内容——是什么,何时,何地完成功能。
引入(Introduction)允许我们向现有的类中添加新方法或者属性。
织入(Weaving)是把切面应用到目标对象并创建新的代理对象的过程,分为编译期织入、类加载期织入和运行期织入。
三、Spring Boot AOP实战
3.1 引入依赖
Spring Boot使用AOP需要添加spring-boot-starter-aop依赖,如下:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
不需要再添加aspectjweaver的依赖了,因为spring-boot-starter-aop包含了aspectjweaver,并且版本是较新的版本,如果在添加老版本(如1.5.4)启动会报错。
3.2 编写Controller
直接定义一个controller,代码如下:
@RestController
public class AopController {
@RequestMapping("/hello/{s}")
@Monitor(value = "speed")
public String sayHello(@PathVariable String s){
System.out.println(s);
System.out.println("hello");
return "ok";
}
}
3.3 定义切面
Spring采用@AspectJ注解对POJO进行标注,该注解表明该类不仅仅是一个POJO,还是一个切面。切面是切点和通知的结合,那么定义一个切面就需要编写切点和通知。在代码中,只需要添加@Aspect注解即可。
3.3.1 定义切点
切点是通过@Pointcut注解和切点表达式定义的。
@Pointcut注解可以在一个切面内定义可重用的切点。
由于Spring切面粒度最小是达到方法级别,而execution表达式可以用于明确指定方法返回类型,类名,方法名和参数名等与方法相关的部件,并且实际中,大部分需要使用AOP的业务场景也只需要达到方法级别即可,因而execution表达式的使用是最为广泛的。如图是execution表达式的语法:
execution表示在方法执行的时候触发。以“”开头,表明方法返回值类型为任意类型。然后是全限定的类名和方法名,“”可以表示任意类和任意方法。对于方法参数列表,可以使用“..”表示参数为任意类型。如果需要多个表达式,可以使用“&&”、“||”和“!”完成与、或、非的操作。
3.3.2 定义通知
通知有五种类型,分别是:
前置通知(@Before):在目标方法调用之前调用通知
后置通知(@After):在目标方法完成之后调用通知
环绕通知(@Around):在被通知的方法调用之前和调用之后执行自定义的方法
返回通知(@AfterReturning):在目标方法成功执行之后调用通知
异常通知(@AfterThrowing):在目标方法抛出异常之后调用通知
代码中定义了三种类型的通知,使用@Before注解标识前置通知,打印“beforeAdvice...”,使用@After注解标识后置通知,打印“AfterAdvice...”,使用@Around注解标识环绕通知,在方法执行前和执行之后分别打印“before”和“after”。这样一个切面就定义好了,代码如下:
@Aspect
@Component
public class AopAdvice {
@Pointcut("execution (* com.dj.example.Controller.*.*(..))")
public void test() {
}
@Before("test()")
public void beforeAdvice() {
System.out.println("beforeAdvice...");
}
@After("test()")
public void afterAdvice() {
System.out.println("afterAdvice...");
}
@Around("test()")
public void aroundAdvice(ProceedingJoinPoint proceedingJoinPoint) {
System.out.println("before");
try {
proceedingJoinPoint.proceed();
} catch (Throwable t) {
t.printStackTrace();
}
System.out.println("after");
}
@AfterReturning(returning = "ret",pointcut = "test()")
public void doAfterReturning(JoinPoint joinPoint, Object ret) throws Throwable {
// 处理完请求,返回内容
Object[] obj = joinPoint.getArgs(); //获取目标方法的参数信息 Object[] 跟列表的区别 定长不可变
System.out.println(Arrays.toString(obj)); //参数列表 [helloTest]
System.out.println(ret.getClass().getName()); //返回的参数
System.out.println(joinPoint.getThis()); //获取代理对象
System.out.println(joinPoint.getTarget()); //获取被代理的对象
System.out.println(joinPoint.getSignature().getDeclaringTypeName()); //类名
System.out.println(joinPoint.getSignature().getName()); //目标方法名
}
}
3.3.3 运行结果
3.3.4 自定义注解
@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface Monitor {
String value() default "";
}
3.3.5 自定义注解使用
@Monitor(value = "speed")
3.3.6 目标方法上使用注解
@Aspect
public class MonitorInterceptor {
@Around("execution(* *..*.*(..)) && @annotation(monitor)")
public void doMonitor (ProceedingJoinPoint joinPoint, Monitor monitor) {
//业务代码
}
}
execution(* ...*(..))表示执行任何方法
@annotation(monitor)表示方法上带有Monitor注解的
当满足这两个条件时即时满足条件的切入点
如果不想使用这种方式,可以考虑使用
@Around("execution(@Monitor * *..*.*(..))")
I can feel you forgetting me。。 有一种默契叫做我不理你,你就不理我