前言:
在前面我们已经粗略的介绍了什么是AOP以及各种基础知识点,而本篇我们将聚焦于AOP的细节,详细的讲解一下AOP中的通知类型,通知顺序,切入点表达式以及连接点。通过对AOP的熟练掌握,我们可以快速编写出低耦合度的代码。
目录
前言:
通知类型:
通知以及切面顺序:
通知顺序从先到后分别为:
切面优先级从前到后分别为:
切入点表达式:
总结:
通知类型:
在前面我们使用的通知类型都是环绕通知
而在AOP中还有很多的通知类型:
-
前置通知(Before Advice):在目标方法执行之前执行的通知。它可以用于执行一些预处理操作,比如安全检查、参数验证等。
-
后置通知(After Advice):在目标方法执行之后(不考虑方法是否出现异常)执行的通知。它可以用于执行一些清理操作,比如释放资源、记录操作日志等。
-
返回通知(After Returning Advice):在目标方法成功执行并返回结果之后执行的通知。可以获取目标方法的返回值,并进行相应的处理。
-
异常通知(After Throwing Advice):在目标方法抛出异常时执行的通知。它可以用于捕获目标方法抛出的异常,并进行相应的处理或记录错误信息。
-
环绕通知(Around Advice):在目标方法执行之前和之后都可以执行的通知。环绕通知可以完全控制目标方法的执行过程,包括参数的传递、调用目标方法、异常的处理等。
这些通知类型可以根据实际需求选择使用,并结合切入点(Pointcut)定义来确定在哪些连接点上触发对应的通知。通过使用各种通知类型,可以实现对横切关注点的精确控制,并在目标对象的核心业务逻辑与横切逻辑之间实现解耦。
@Around环绕通知需要自己调用 ProceedingJoinPoint.proceed()来让原始方法执行,而其他的通知不需要考虑让目标方法执行
@Around环绕通知的返回值,必须指定为object类来接收原始方法的返回值
通知以及切面顺序:
如果有多个切面的切入点都匹配到了目标方法,那么在目标方法运行的时候,多个通知方法都会执行,那么就会产生一个先后顺序
通知顺序从先到后分别为:
-
环绕通知(Around Advice):环绕通知的执行优先于其他通知。它包裹着目标方法的执行,并可以在方法执行前后进行额外的处理。环绕通知可以决定是否调用目标方法,也可以自定义目标方法的输出结果。
-
前置通知(Before Advice):前置通知在目标方法执行前被调用。它可以用于执行一些预处理操作,例如参数校验、安全检查等。多个前置通知按照配置的先后顺序执行。
-
后置通知(After Advice):后置通知在目标方法执行后(不考虑方法是否出现异常)被调用。它可以用于执行一些清理操作,比如资源释放、记录操作日志等。多个后置通知按照配置的先后顺序执行。
-
返回通知(After Returning Advice):返回通知在目标方法成功执行并返回结果后被调用。它可以获取目标方法的返回值,并进行相应的处理。多个返回通知按照配置的先后顺序执行。
-
异常通知(After Throwing Advice):异常通知在目标方法抛出异常后被调用。它可以捕获目标方法抛出的异常,并进行相应的处理或记录错误信息。多个异常通知按照配置的先后顺序执行。
需要注意的是,同一个切面中的多个通知按照配置的先后顺序执行,而不同切面之间的通知执行顺序取决于切面的优先级。可以通过AOP框架的配置来控制通知的顺序。
切面优先级从前到后分别为:
-
注解配置优先级高:如果切面使用注解配置,并且配置了优先级属性,则优先级高于其他方式设置的切面。
-
显式设置优先级高:如果通过编程方式或XML配置显式设置了切面的优先级属性,优先级高于默认设置的切面。
-
默认设置:如果没有显式设置切面的优先级,AOP框架通常会给予切面一个默认的优先级顺序,例如按照切面加载的顺序确定优先级。
使用切面的优先级可以确保对同一个连接点的处理顺序,从而避免不同切面之间的冲突或干扰。
切入点表达式:
- 切入点表达式:描述切入点方法的一种表达式。
- 作用:主要用来决定项目中的哪些方法需要接入通知。
常见形式:
1.execution(......):根据方法的签名来匹配
@Around("execution(public void com.example.MyClass.myMethod(..))")
public void aroundMyMethod(ProceedingJoinPoint joinPoint) throws Throwable {
}
2.@annotation(......):根据注解进行匹配
@Around("@annotation(com.example.MyAnnotation)")
public void aroundAnnotatedMethods(ProceedingJoinPoint joinPoint) throws Throwable {
}
通配符描述切入点:
-
*
通配符:表示匹配零个或多个字符。在AOP中,*
通配符可以用于匹配任意数量的任意字符。例如,com.example.*Service
可以匹配任何以 “Service” 结尾的类名,并位于 “com.example” 包下的类。 -
..
通配符:表示匹配任意多层级的目录或包。在AOP中,..
通配符可以用于匹配任意多层级的目录路径或包路径。例如,com.example..*Service
可以匹配任何以 “Service” 结尾的类,并位于 “com.example” 包及其子包下的类。
扩展切入点表达式功能:
切入点表达式可以使用类似于正则表达式的语法,将上述元素组合起来进行更精确的匹配。同时,AOP框架也提供了一些特殊的语法和关键词来扩展切入点表达式的功能,例如:
-
&&:逻辑与操作符,用于将多个关键元素连接起来,表示需要同时满足多个条件。
-
||:逻辑或操作符,用于将多个关键元素连接起来,表示满足任意一个条件即可。
-
!:逻辑非操作符,用于对关键元素的匹配结果取反。
-
@target:用于匹配目标类上应用了特定注解的连接点。
-
@within:用于匹配目标类及其子类上应用了特定注解的连接点。
切入点表达式的准确使用可以帮助开发人员精确地选择目标连接点,并将切面逻辑应用到指定的位置,避免不必要的拦截和干扰,提高AOP的效率和精度。
总结:
本文我们对AOP中的一些知识点进行了深入解析,而这些知识点在实际使用中的利用程度很高,因此我们要熟练的掌握这些知识点。
如果我的内容对你有帮助,请点赞,评论,收藏。创作不易,大家的支持就是我坚持下去的动力!