目录
一、为什么需要面向切面编程?
二、Spring AOP术语
三、Spring Boot AOP实现
3.1 引入依赖
3.2 编写用于拦截的bean
3.3 定义切面
3.3.1 定义切点
3.3.2 定义通知
3.3.3 前置、后置、返回、异常通知的切入:
3.3.4 环绕通知
面向对象编程(OOP)的好处是显而易见的,缺点也同样明显。当需要为多个不具有继承关系的对象添加一个公共的方法的时候,例如日志记录、性能监控等,如果采用面向对象编程的方法,需要在每个对象里面都添加相同的方法,这样就产生了较大的重复工作量和大量的重复代码,不利于维护。面向切面编程(AOP)是面向对象编程的补充,简单来说就是统一处理某一“切面”的问题的编程思想。如果使用AOP的方式进行日志的记录和处理,所有的日志代码都集中于一处,不需要再每个方法里面都去添加,极大减少了重复代码。
1、通知(Advice):就是我们上面提到的除了业务代码本身之外,一些“可重用的一些代码”,先定义好,在需要的地方进行重用;
2、连接点(JoinPoint):允许你使用“通知”的地方,就是允许你重复使用代码的地方,方法执行的前后或者方法抛出异常时,Spring只支持方法级别的连接点;
3、切入点(Pointcut): 假如有若干方法,但是我们并不是想在所有的方法中“重用这段可复用的代码”,我们只想在某些特定的方法上使用通知,那么我们就可以使用切入点来进行这些连接点的“筛选了”;
4、切面(Aspect):切面简单来讲就是切入点(JoinPoint)和通知(Advice)的结合体。通知(Advice)决定了要干什么(通知的方法体)?在什么时候干?(定义通知的注解类型)。而切入点决定了要在哪儿干(即执行通知定义的方法体)。
5、引入(Introduction):允许我们向目标对象添加新的方法属性(即通过执行通知来控制对目标方法的访问);
6、目标对象(Target):引入中所提到的目标对象,也就是要被通知的对象,也就是执行真正的业务逻辑,可以在毫不知情的情况下,织入我们的切面;
7、代理(Proxy):Spring中的AOP都是通过动态代理来实现的;
8、织入(Weaving):把切面应用到目标对象,创建代理对象的过程;
Spring Boot使用AOP需要添加spring-boot-starter-aop依赖,如下:
org.springframework.boot
spring-boot-starter-aop
直接定义一个controller,代码如下:
@RestController
public class UserController {
@RequestMapping("/")
public String hello(){
System.out.println("我就是被切入的目标对象的方法!");
return "";
}
}
Spring采用@Aspect注解对目标类进行标注,该注解表明该类不仅仅是一个目标类,还是一个切面。切面是切点和通知的结合,那么定义一个切面就需要编写切点和通知。在代码中,不仅需要添加@Aspect注解还需要添加@Component组件注解,表示被扫描并提交给Spring管理。
切点是通过@Pointcut注解和切点表达式定义的。
@Pointcut注解可以在一个切面内定义可重用的切点。
由于Spring切面粒度最小是达到方法级别,而execution表达式可以用于明确指定方法返回类型,类名,方法名和参数名等与方法相关的部件,并且实际中,大部分需要使用AOP的业务场景也只需要达到方法级别即可,因而execution表达式的使用是最为广泛的。如图是execution表达式的语法:
execution表示在方法执行的时候触发。以“*”开头,表明方法返回值类型为任意类型。然后是全限定的类名和方法名,“*” 可以表示任意类和任意方法。对于方法参数列表,可以使用“..”表示参数为任意类型。如果需要多个表达式,可以使用“&&”、“||”和“!”完成与、或、非的操作。
通知有五种类型,分别是:
前置通知(@Before):在目标方法调用之前调用通知
后置通知(@After):在目标方法完成之后调用通知
返回通知(@AfterReturning):在目标方法成功执行之后调用通知
异常通知(@AfterThrowing):在目标方法抛出异常之后调用通知
环绕通知(@Around):在被通知的方法调用之前和调用之后执行自定义的方法
环绕通知中可以实现其他所有通知,所以我们用代码来分别显示:
@Component
@Aspect
public class AnnotationAOP {
@Pointcut("execution(* com.songqiao.controller.UserController.hello())")
public void PointCut(){}
//通过JoinPoint接口可以获取到方法名
@Before("PointCut()")
public void beforeAOP(JoinPoint joinPoint){
String methodName = joinPoint.getSignature().getName();
System.out.println("前置通知已执行!被切入的方法名是"+methodName);
}
@After("PointCut()")
public void afterAOP(){
System.out.println("后置通知已执行!");
}
@AfterReturning("PointCut()")
public void afterReturningAOP(){
System.out.println("返回通知已执行!");
}
//通过throwing属性可以获取到异常信息
@AfterThrowing(value = "PointCut()",throwing = "ex")
public void afterThrowingAOP(Throwable ex){
System.out.println("异常通知!异常信息为:"+ex);
}
}
当我们的切入的方法产生异常时,会执行切入的异常通知并通过 throwing属性获取异常信息并输出,由于发生异常,方法没有执行return 语句,所以返回通知不会执行。请看效果:
测试:
当我们把异常报错修改后即可执行返回通知。所以说返回通知与异常通知只能执行一个!
环绕通知就是AOP的动态代理模式,可以封装以上所有的通知。
@Component
@Aspect
public class AnnotationAOP {
@Pointcut("execution(* com.songqiao.controller.UserController.hello())")
public void PointCut(){}
@Around("PointCut()")
public Object aroundAdviceMethod(ProceedingJoinPoint joinPoint){
Object result=null;
try {
//前置通知位置
System.out.println("环绕的前置通知");
//相当于是目标对象方法的执行
result = joinPoint.proceed();
//返回通知位置
System.out.println("环绕的返回通知");
} catch (Throwable throwable) {
//异常通知位置
throwable.printStackTrace();
System.out.println("环绕的异常通知");
}finally {
//后置通知位置
System.out.println("环绕的后置通知");
}
return result;
}
}
测试:
希望本篇文章能够帮助到大家更熟练的掌握面向切面编程。