当前位置: 首页 > news >正文

Spring原理揭秘--Spring的AOP

在这之前我们已经介绍了AOP的基本功能和概念,那么当AOP集成到spring则会发生改变。

Spring AOP 中的Joinpoint:

之前提高了很多Joinpoint的类型,但是在spring中则只会有方法级别的Joinpoint,像构造方法,字段的调用都没适配。原因是Spring框架一直追求轻量级和高效,而仅仅采用方法级的Joinpoint即可满足百分之八十的需求。同时如果采用类中的属性级别的Joinpoint则会破坏对象的封装。如果需求非常特殊超过了拿百分之八十的需求那么不妨可以使用AspectJ的方式。

Spring AOP 中的Pointcut:

Spring中以接口定义Pointcut作为其最顶层的抽象接口,该接口定义了两个方法用来帮助捕获系统中相应的Joinpoint

两个方法的ClassFilter和MethodMatcher分别是用来匹配将执行织入操作的对象以及相应的方法。

也就是说ClassFilter是对于类型对象的匹配而MethodMatcher则是对于方法级别的匹配,

MethodMatcher的复杂度相比于ClassFilter要高。原因是ClassFilter仅仅只需要匹配类型即可而MethodMatcher则可能需要匹配方法名称或者方法名称+参数

因此对于MethodMatcher则有多个方法进行匹配

有两个matches的方法,而这两个方法的分界线就是isRuntime方法。在对具体的方法进行拦截的时候,可以忽略每次方法执行的时候调用者传入的参数,也可以每次都检查这些方法调用参数,以强化拦截条件,都是与isRuntime的返回值有关如果返回值为false则会触发第一个matches方法这种属于不会考虑具体方法的参数,而如果返回结果为true则会调用第二个matches方法表示采用参数的匹配

将这两个MethodMatcher类型分为StaticMethodMatcher和DynamicMethodMatcher类型。StaticMethodMatcher类型则是无参数检查,这种无需参数检查的类型可以缓存在框架内部,同时性能比较出色,对于DynamicMethodMatcher类型是参数检查,因此不会进行缓存操作,同时性能上相比StaticMethodMatcher则是损耗很大。

常见的PointCut

下面我们来介绍几个常见的PointCut

NameMatchMethodPointcut:

这个是最简单的Pointcut实现,属于StaticMethodMatcher的子类,可以根据自身指定的一组方法名称与Joinpoint处的方法的名称进行匹配

JdkRegexpMethodPointcut和Perl5RegexpMethodPointcut

StaticMethodMatcher的子类有一个专门基于正则表达式的实现分支,而JdkRegexpMethodPointcut和Perl5RegexpMethodPointcut则是不同类型的正则表达式的具体实现。

AnnotationMatchingPointcut

根据目标对象中是否存在指定类型的注解来匹配Joinpoint,要使用该类型的Pointcut首先需要声明相应的注解。

CompossablePointcut

提供逻辑运算功能,而ComposablePointcut就是Spring AOP提供的可以进行Pointcut逻辑运算的Pointcut实现。它可以实现Pointcut之间的并以及交运算。

ControlFlowPointcut

ControlFlowPointcut则是最特殊的Pointcut类型,在理解和使用上都需要我们多付出点脑细胞,ControlFlowPointcut匹配程序的调用流程,不是对某个方法执行所在的Joinpoint处的单一特征进行匹配。通过ControlFlowPointcut我们可以实现指定只有当某个类的方法在被执行的类对象调用的时候才会对方法进行拦截处理

Spring AOP中的Advice

Advice实现了将被织入到Pointcut规定的Joinpoint处的横切逻辑。在spring中,advice按照其自身实例能否在目标对象类的所有实例中共享这一标准,可以划分为两大类,即per-class类型的advice和per-instance类型的advice。

per-class类型的Advice

per-class类型的Advice是指该类型的实例可以在目标对象类的所有实例之间进行共享,这种类型的Advice只是提供方法拦截的功能,不会为目标对象类保存任何状态或者添加新的特性。

那么per-class有着以下几种类型

Before advice

实现的横切逻辑将在相应的Joinpoint之前执行,在Before advice执行完成之后,程序执行流程将从Joinpoint处继续执行,所以Before Advice通常不会打断程序执行的流程,但是如果有必要,也可以通过抛出相应异常的形式中断程序流程

ThrowsAdvice

throwsAdvice通常用于对系统重特定的异常情况的监控,以统一的方式对所发生的异常进行处理,一旦捕获到异常,需要马上以某种方式通知系统的监控或者运营人员

AfterReturningAdvice

通过spring的AfterReturningAdvice,我们可以访问当前joinpoint的方法返回值,方法,方法参数以及所在的目标对象

Around Advice

Spring AOP没有提供AfterAdvice,使得我们没有一个合适的advice类型来承载类似于系统资源清除之类的横切逻辑。Spring AOP的AfterReturningAdvice不能更改Joinpoint所在方法的返回值,使得我们在方法正常返回后无法对其进行更多的敢于。有了Around Advice则可以解决这个问题

per-instance类型的Advice

在Spring AOP当中只有一个类型的per-instance型Advice就是Introduction

Introduction

Introduction允许你在不修改目标类代码的情况下,为其动态添加新的接口和实现。这相当于在运行时为目标对象 “混入” 额外的行为,使其具备原本没有的功能。

// 定义新接口
public interface Monitorable {void setMonitorActive(boolean active);boolean isMonitorActive();
}// 实现接口和 IntroductionInterceptor
public class MonitorInterceptor implements MethodInterceptor, Monitorable {private boolean monitorActive = false;@Overridepublic Object invoke(MethodInvocation invocation) throws Throwable {if (monitorActive && invocation.getMethod().getName().startsWith("get")) {System.out.println("Monitoring: " + invocation.getMethod().getName());}return invocation.proceed();}@Overridepublic void setMonitorActive(boolean active) {this.monitorActive = active;}@Overridepublic boolean isMonitorActive() {return monitorActive;}
}// 创建 IntroductionAdvisor
public class MonitorAdvisor extends DefaultIntroductionAdvisor {public MonitorAdvisor() {super(new MonitorInterceptor(), Monitorable.class);}
}
增强类型作用是否改变类结构
Before/After在方法前后添加行为
Around环绕方法执行,完全控制调用过程
Introduction为目标对象添加新接口和实现是(运行时)
http://www.dtcms.com/a/284260.html

相关文章:

  • cuda编程笔记(8)--线程束warp
  • Cookie 与 Session概述
  • AI编程实战:如何让AI生成带参数和返回值的Python函数——以PDF文本提取为例
  • 【橘子分布式】gRPC(理论篇)
  • 要实现在调用  driver.get()  后立即阻止页面自动跳转到 Azure 登录页,可通过以下几种方法实现:
  • Redis完全指南:从基础到实战(含缓存问题、布隆过滤器、持久化及Spring Boot集成)
  • 前端 cookie 使用
  • 独家|理想汽车放弃华为PBC模式,回归OKR理想汽车
  • 自动化测试工具 Selenium 入门指南
  • 大带宽服务器对于高流量网站的作用
  • Kubernetes v1.33:容器生命周期管理的重要演进
  • 断网情况下,网线直连 Windows 笔记本 和Ubuntu 服务器
  • python的抗洪救灾管理系统
  • 分布式系统中脑裂问题
  • 数据结构入门 (二):挣脱连续空间的束缚 —— 单向链表详解
  • LiteCloud超轻量级网盘项目基于Spring Boot
  • 大模型AI制作svg流程图过分简单粗糙单调的问题及修改调整方法
  • AI大模型教程 Langchain AI原生应用开发 Milvus AnythingLLM Dify 仿京东《京言》AI实战解析
  • 【赵渝强老师】Redis的主从复制集群
  • mongodb-org-server_8.0.11_amd64.deb 这个文件怎么安装
  • Java爬虫与正则表达式——用正则来爬取数据
  • 二刷 黑马点评 秒杀优化
  • 板凳-------Mysql cookbook学习 (十二--------1)
  • 医院各类不良事件上报,PHP+vscode+vue2+element+laravel8+mysql5.7不良事件管理系统源代码,成品源码,不良事件管理系统
  • React事件处理
  • 【FFmpeg 快速入门】本地播放器 项目
  • c++:explicit关键字
  • Python枚举技巧:轻松获取索引与值
  • 【Linux手册】缓冲区:深入浅出,从核心概念到实现逻辑
  • Python爬虫入门到实战(2)-selenium驱动浏览器