Spring源码04 - AOP深入设计原理

devtools/2025/1/23 17:03:21/

AOP深入设计原理

文章目录

  • AOP深入设计原理
    • 一:JVM下的AOP
      • 1:Java程序运行在JVM中的特征
      • 2:Java程序执行流
      • 3:引入了代理模式的Java程序执行流
    • 二:Spring AOP工作原理

本文整理自csdn博主亦山 -> 《Spring设计思想》AOP设计基本原理

https://blog.csdn.net/luanlouis/article/details/51095702?spm=1001.2014.3001.5501

一:JVM下的AOP

1:Java程序运行在JVM中的特征

当我们在某个类Foo中写好了一个main()方法,然后执行java Foo,Java程序就开启了,如下:

java">public class Foo {public static void main(String[] args) {// your code}
}

在上述过程中,JVM到底帮助我们做了什么

当你执行java Foo 的时候,JVM会创建一个主线程main,这个主线程以上述的main()方法作为入口,开始执行你的代码。

每一个线程在内存中都会维护一个属于自己的栈(Stack),记录着整个程序执行的过程。

栈里的每一个元素称为栈帧(Stack Frame),栈帧表示着某个方法调用,会记录方法调用的信息;

实际上我们在代码中调用一个方法的时候,在内存中就对应着一个栈帧的入栈和出栈【虚拟机栈 - 线程私有 & 方法调用】

在某个特定的时间点,一个Main线程内的栈会呈现如下图所示的情况:

在这里插入图片描述
从线程栈的角度来看,我们可以看到,JVM处理Java程序的基本单位是**方法调用**。【宏观角度】

实际上,JVM执行的最基本单位的指令(即原子操作)是汇编语言性质的机器字节码。这里是宏观角度

我们如何能够获取到JVM线程栈中的方法调用的内容呢??

当我们的代码抛出异常而未捕获或者运行时出现了Error错误时,我们会受到一个非常讨厌的Log信息:

在这里插入图片描述
当然,除了代码抛出异常外,我们还是可以其他方式察觉JVM线程栈内的内容。

可以通过Thread.dumpStack()方法创建一个假的Exception实例,然后将这个Exception实例记录的当前线程栈的内容输出到标准错误流中。

例如我在某处代码里执行了Thread.dumpStack()方法

在这里插入图片描述

2:Java程序执行流

了解AOP、连接点(Join Point)、切入点(point cut) 的概念

如果从虚拟机线程栈的角度考虑Java程序执行的话,那么,你会发现,真个程序运行的过程就是方法调用的过程。

我们按照方法执行的顺序,将方法调用排成一串,这样就构成了Java程序流
在这里插入图片描述
基于时间序列,我们可以将方法调用排成一条线。而每个方法调用则可以看成Java执行流中的一个节点。这个节点在AOP的术语中就是连接点Join Point

一个Java程序的运行的过程,就是若干个连接点连接起来依次执行的过程

通常面向对象(OOP)的程序,代码都是按照时间序列纵向展开的,而他们都有一个共性:

即都是已方法调用作为基本执行单位展开的。将方法调用当做一个连接点,那么由连接点串起来的程序执行流就是整个程序的执行过程。

而AOP(Aspect Oriented Programming)则是从另外一个角度来考虑整个程序的:

AOP将每一个方法调用,即连接点作为编程的入口,针对方法调用进行编程。

从执行的逻辑上来看,相当于在之前纵向的按照时间轴执行的程序横向切入。相当于将之前的程序横向切割成若干的面,即Aspect切面

在这里插入图片描述
既然AOP是针对切面进行的编程的,那么,你需要选择哪些连接点Joint Point作为你的编程对象呢?

因为切面本质上是每一个方法调用,选择切面的过程实际上就是选择方法的过程。

那么,被选择的切面(Aspect)在AOP术语里被称为切入点(Point Cut). 切入点实际上也是从所有的连接点挑选自己感兴趣的连接点的过程

在这里插入图片描述
既然AOP是针对方法调用(连接点)的编程,现在又选取了你感兴趣的自己感兴趣的链接点—切入点(Point Cut)了

那么,AOP能对它做什么类型的编程呢?AOP能做什么呢?

这个之前,先要知道一个非常重要的问题: 既然AOP是对方法调用进行的编程,那么,AOP如何捕获方法调用的呢?

弄清楚这个问题,就不得不了解设计模式中的代理模式了

3:引入了代理模式的Java程序执行流

AOP实现的机制

假设在我们的Java代码里,都为实例对象通过代理模式创建了代理对象,访问这些实例对象必须要通过代理

那么,加入了proxy对象的Java程序执行流会变得稍微复杂起来

在这里插入图片描述
由上图可以看出,只要想调用某一个实例对象的方法时,都会经过这个实例对象相对应的代理对象, 即执行的控制权先交给代理对象。

代理模式属于Java代码中经常用到的、也是比较重要的设计模式。

代理模式可以为某些对象除了实现本身的功能外,提供一些额外的功能,大致作用如下图所示:

在这里插入图片描述
加入了代理模式的Java程序执行流,使得所有的方法调用都经过了代理对象。

对于Spring AOP框架而言,它负责控制着真个容器内部的代理对象。

当我们调用了某一个实例对象的任何一个非final的public方法时,整个Spring框架都会知晓。【上帝视角】

既然Spring代理层可以察觉到你所做的每一次对实例对象的方法调用,那么,Spring就有机会在这个代理的过程中插入Spring的自己的业务代码。

在这里插入图片描述

二:Spring AOP工作原理

AOP能对切入点做什么样的编程呢?

我们先将代理模式下的某个连接点细化,会看到如下过程:真实的Spring AOP的proxy角色扮演的只能比这复杂的多,这里只是简化

在这里插入图片描述
可见代理模式的代理角色最起码要考虑三个阶段:

  1. 在调用真正对象的方法之前,应该需要做什么?
  2. 在调用真正对象的方法过程中,如果抛出了异常,需要做什么?
  3. 在调用真正对象的方法后,返回了结果,需要做什么?

而AOP对这个方法调用的编程,就是针对这三个阶段插入自己的业务代码。

现在我们假设

  • 当前RealSubject这个角色的类是 org.luanlouis.springlearning.aop.FooService

  • 当前这个连接点对应的方法签名是:public void foo()。

那么上述的代理对象的三个阶段将会有以下的处理逻辑:

  • 在调用真正对象的方法之前
    • proxy会告诉Spring AOP:“我将要调用类org.luanlouis.springlearning.aop.FooService的public void foo() ,在调用之前,你有什么处理建议吗?”;
    • Spring AOP这时根据proxy提供的类名和方法签名,然后拿这些信息尝试匹配是否在其感兴趣的切入点内,如果在感兴趣的切入点内,Spring AOP会返回MethodBeforeAdvice处理建议,告诉proxy应该执行的操作;
  • 在调用真正对象的方法过程中,如果抛出了异常
    • proxy告诉Spring AOP:“我调用类org.luanlouis.springlearning.aop.FooService的public void foo()过程中抛出了异常,你有什么处理建议?”
    • Spring AOP这时根据proxy提供的类名和方法签名,然后拿这些信息尝试匹配是否在其感兴趣的切入点内,如果在感兴趣的切入点内,Spring AOP会返回ThrowsAdvice处理建议,告诉proxy应该执行的操作;
  • 在调用真正对象的方法后,返回了结果了
    • proxy告诉Spring AOP:“我调用类org.luanlouis.springlearning.aop.FooService的public void foo()结束了,你有什么处理建议?”
    • Spring AOP这时根据proxy提供的类名和方法签名,然后拿这些信息尝试匹配是否在其感兴趣的切入点内,如果在感兴趣的切入点内,Spring AOP会返回AfterReturingAdivce处理建议,告诉proxy应该执行的操作;

在这里插入图片描述
上述的示意图中已经明确表明了Spring AOP应该做什么样的工作:根据proxy提供的特定类的特定方法执行的特定时期阶段给出相应的处理建议。

要完成该工作,Spring AOP应该实现:

  • 确定自己对什么类的什么方法感兴趣? --即确定 AOP的切入点(Point Cut),这个可以通过切入点(Point Cut)表达式来完成
  • 对应的的类的方法的执行特定时期给出什么处理建议?------这个需要Spring AOP提供相应的建议 ,即我们常说的Advice

在这里插入图片描述

再次声明,将代理模式拆成三个部分是简化AOP的原理,实际上AOP的处理过程比这个复杂的多


http://www.ppmy.cn/devtools/152935.html

相关文章

每日一题——二分法求旋转数组的最小数字

二分法求旋转数组的最小数字 描述数据范围:要求: 示例示例 1示例 2 解题思路算法流程 代码实现关键点解释举例分析示例 1:[3, 4, 5, 1, 2]示例 2:[3, 100, 200, 3] 总结 描述 有一个长度为 n 的非降序数组,例如 [1, 2…

考研408笔记之数据结构(三)——串

数据结构(三)——串 1. 串的定义和基本操作 本节内容很少,重点是串的模式匹配,所以对于串的定义和基本操作,我就简单提一些易错点。另外,串也是一种特殊的线性表,只不过线性表是可以存储任何东…

RK3568笔记七十六:使用V4L2框架录制MP4视频保存到本地

若该文为原创文章,转载请注明原文出处。 录制MP4使用的是ffmpeg,如何编译自行处理。 使用的是正点原子的RK3568测试,其他板子自行调试。 一、程序功能介绍 说明: 程序参考FFMPEG提供的例子程序muxing.c进行修改。 功能介绍: 程序里目前有一个子线程和一个主线程,子线程…

SSM项目本地Tomcat部署

目录 1、打包 2、部署在本地Tomcat上 3、运行tomcat(startup) 1、打包 在生命周期中,完成打包。 注意:打包时会测试,测试时可能会测试根据id删除。第二次的测试就会出错,导致打包失败。 从target目录下…

【趣学SQL】第三章:数据处理与管理 3.1数据清洗技术——给数据库做“数据SPA“的魔幻之旅

第三章:数据处理与管理 3.1 数据清洗技术——给数据库做"数据SPA"的魔幻之旅 欢迎来到「数据库美容院」!今天我们将化身"数据美容师",用一家虚拟网红餐厅的翻车案例,教你如何把脏乱差的原始数据变成清爽整洁…

网站HTTP改成HTTPS

您不仅需要知道如何将HTTP转换为HTTPS,还必须在不妨碍您的网站自成立以来建立的任何搜索排名权限的情况下进行切换。 为什么应该从HTTP转换为HTTPS? 与非安全HTTP于不同,安全域使用SSL(安全套接字层)服务器上的加密代…

nuxt3项目打包部署到服务器后配置端口号和开启https

nuxt3打包后的项目部署相对于一般vite打包的静态文件部署要稍微麻烦一些,还有一个主要的问题是开发环境配置的.env环境变量在打包后部署时获取不到,具体的解决方案可以参考我之前文章 nuxt3项目打包后获取.env设置的环境变量无效的解决办法。 这里使用的…

Java菜鸟养成计划(java基础)--java数据类型

数据类型 1、什么是数据类型?2、java中的数据类型有哪些?3、基本数据类型有哪些?3.1、布尔类型(boolean)3.2、字符类型3.3、整形(默认int)3.4、浮点型【默认double】小数3.5、基本数据类型之间的…