深度理解Spring----AOP

查看了很多资料,浏览了很多文章,这里总结一下AOP,启蒙原文: https://blog.csdn.net/q982151756/article/details/80513340.

一、对AOP的初印象

首先先给出一段比较专业的术语(来自百度):

在软件业,AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期间动态代理实现程序功能的统一维护的一种技术。AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。

主要功能:日志记录,性能统计,安全控制,事务处理,异常处理等等。

然后我们举一个易于理解的例子:

我们一般做活动的时候,一般对每一个接口都会做活动的有效性校验(是否开始、是否结束等等)、以及这个接口是不是需要用户登录。

按照正常的逻辑,我们可以这么做。

在这里插入图片描述
这有个问题就是,有多少接口,就要多少次代码copy。对于一个“懒人”,这是不可容忍的。好,提出一个公共方法,每个接口都来调用这个接口。这里有点切面的味道了。
在这里插入图片描述
同样有个问题,我虽然不用每次都copy代码了,但是,每个接口总得要调用这个方法吧。于是就有了切面的概念,我将方法注入到接口调用的某个地方(切点)。
在这里插入图片描述
这样接口只需要关心具体的业务,而不需要关注其他非该接口关注的逻辑或处理。

红框处,就是面向切面编程。

  • 面向切面编程,往往被定义为促使软件系统实现关注点分离的技术。系统是由许多不同的组件所组成的,每一个组件负责一块特定功能。除了实现自身核心功能之外,这些组件还经常承担着额外的职责。例如日志、事务管理和安全等,这样的服务经常融入到具有核心业务逻辑的组件中去。这些系统服务经常被称为横切关注点,因为它们会跨越系统的多个组件。
  • AOP就是把与核心业务逻辑无关的代码全部抽取出来,放置到某个地方集中管理,然后在具体运行时,再由容器动态织入这些共有代码。

这种在运行时,动态地将代码切入到类的指定方法、指定位置上的编程思想就是面向切面的编程。

AOP把软件系统分为两个部分:核心关注点和横切关注点。

业务处理的主要流程是核心关注点,与之关系不大的部分是横切关注点。

横切关注点的一个特点是,经常发生在核心关注点的多处,而各处都基本相似。比如权限认证、日志、事务处理。

Aop 的作用在于分离系统中的各种关注点,将核心关注点和横切关注点分离开来。正如Avanade公司的高级方案构架师Adam Magee所说,AOP的核心思想就是“将应用程序中的业务逻辑同对其提供支持的通用服务进行分离”。

AOP相关术语

  • 通知(Advice)
    就是你想要的功能,比如 安全,事物,日志等。你先定义好这些东西,然后在想用的地方使用。

  • 连接点(Join Point)
    就是spring允许你使用通知的地方,那可真就多了,基本每个方法的前、后(两者都有也行),或抛出异常时都可以是连接点,spring只支持方法连接点。其他如aspectJ,还可以让你在构造器或属性注入时都行,不过那不是咱关注的,只要记住,和方法有关的前前后后(抛出异常),都是连接点。

  • 切点(Point cut)
    用于定义通知应该切入到哪些连接点上,不同的通知通常需要切入到不同的连接点上。它表示一组 joint point,这些 joint point 或是通过逻辑关系组合起来,或是通过通配、正则表达式等方式集中起来,它定义了相应的 Advice 将要发生的地方。

  • 切面(Aspect)
    切面是 通知切入 点的结合。现在发现了吧,没连接点什么事情,连接点就是为了让你好理解切点,通知说明了干什么和什么时候干(什么时候通过方法名中的before,after,around等就能知道),而切入点说明了在哪干(指定到底是哪个方法),这就是一个完整的切面定义。

  • 引入(introduction)
    允许我们向现有的类添加新方法属性。就是把切面(也就是通知定义的新方法属性)用到目标类中

  • 目标(target)
    引入中所提到的目标类,也就是要被通知的对象,也就是真正的业务逻辑

  • 织入(weaving)
    把切面应用到目标对象来创建新的代理对象的过程。这个过程可以发生在编译期、类装载期及运行期,当然不同的发生点有着不同的前提条件。譬如发生在编译期的话,就要求有一个支持这种AOP实现的特殊编译器;发生在类装载期,就要求有一个支持AOP实现的特殊类装载器;只有发生在运行期,则可直接通过Java语言的反射机制与动态代理机制来动态实现。

  • 代理对象(Proxy)
    将通知应用到目标对象之后被动态创建的对象。可以简单地理解为,代理对象的功能等于目标对象的核心业务逻辑功能加上共有功能。代理对象对于使用者而言是透明的,是程序运行过程中的产物。

趣味小例子

下面我以一个简单的例子来比喻一下 AOP 中 Aspect, Join point, PointcutAdvice 之间的关系。

让我们来假设一下,从前有一个叫爪哇的小县城,在一个月黑风高的晚上,这个县城中发生了命案。
作案的凶手十分狡猾,现场没有留下什么有价值的线索。
不过万幸的是, 刚从隔壁回来的老王恰好在这时候无意中发现了凶手行凶的过程, 但是由于天色已晚, 
加上凶手蒙着面, 老王并没有看清凶手的面目, 只知道凶手是个男性, 身高约七尺五寸. 
爪哇县的县令根据老王的描述, 对守门的士兵下命令说: 凡是发现有身高七尺五寸的男性, 都要抓过来审问. 
士兵当然不敢违背县令的命令, 只好把进出城的所有符合条件的人都抓了起来。

来让我们看一下上面的一个小故事和 AOP 到底有什么对应关系.

首先我们知道, 在 Spring AOP 中 Join point 指代的是所有方法的 执行点, 而 point cut 是一个描述信息 , 它修饰的是 Join point, 通过 point cut, 我们就可以确定哪些 Joint point 可以被织入 Advice.

对应到我们在上面举的例子, 我们可以做一个简单的类比, Join point 就相当于 爪哇的小县城里的百姓,point cut 就相当于 老王所做的指控, 即凶手是个男性, 身高约七尺五寸, 而 Advice 则是施加在符合老王所描述的嫌疑人的动作: 抓过来审问.

Join point : 爪哇的小县城里的百姓: 因为根据定义, Join point 是所有可能被织入 Advice 的候选的点, 在 Spring AOP中, 则可以认为所有方法执行点都是 Join point. 而在我们上面的例子中, 命案发生在小县城中, 按理说在此县城中的所有人都有可能是嫌疑人.

Point cut :男性, 身高约七尺五寸: 我们知道, 所有的方法(join point) 都可以织入 Advice, 但是我们并不希望在所有方法上都织入 Advice, 而 Point cut 的作用就是提供一组规则来匹配join point, 给满足规则的 join point 添加 Advice. 同理, 对于县令来说, 他再昏庸, 也知道不能把县城中的所有百姓都抓起来审问, 而是根据凶手是个男性, 身高约七尺五寸, 把符合条件的人抓起来. 在这里 凶手是个男性, 身高约七尺五寸 就是一个修饰谓语, 它限定了凶手的范围, 满足此修饰规则的百姓都是嫌疑人, 都需要抓起来审问.

Advice :抓过来审问, Advice 是一个动作, 即一段 Java 代码, 这段 Java 代码是作用于 point cut 所限定的那些 Join point 上的. 同理, 对比到我们的例子中, 抓过来审问 这个动作就是对作用于那些满足 男性, 身高约七尺五寸 的爪哇的小县城里的百姓.

Aspect:Aspect 是 point cut 与 Advice 的组合, 因此在这里我们就可以类比: “根据老王的线索, 凡是发现有身高七尺五寸的男性, 都要抓过来审问” 这一整个动作可以被认为是一个 Aspect.

在这里插入图片描述

Advice类型

通知(Advice)是对目标切入点进行增强的内容。

Spring为通知(Advice)提供了Advice接口,Spring通知按照在目标类方法的连接点位置,可分为以下几类:

before advice: 在 join point 前被执行的 advice. 虽然 before advice 是在 join point 前被执行, 但是它并不能够阻止 join point 的执行, 除非发生了异常(即我们在 before advice 代码中, 不能人为地决定是否继续执行 join point 中的代码)

after return advice: 在一个 join point 正常返回后执行的 advice

after throwing advice: 当一个 join point 抛出异常后执行的 advice

after(final) advice: 无论一个 join point 是正常退出还是发生了异常, 都会被执行的 advice.

around advice: 在 join point 前和 joint point 退出后都执行的 advice. 这个是最常用的 advice.

introduction: introduction可以为原有的对象增加新的属性和方法。

问题:
AOP原理
Spring AOP的使用

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值