Comparing Spring AOP and AspectJ

时间:2020-12-08 17:31:08

AOP 概念

在我们开始之前 , 让我们做一个快速、高级别审查的核心术语和概念 :

  • 方面 — —标准 / 特征代码被分散在多个场所中的应用 , 通常不同于实际的业务逻辑 (例如 , 交易管理) 。各方面集中于特定的切割功能
  • Joinpoint –它的具体执行过程中程序的执行等方法 , 调用构造函数或赋值域
  • 建议采取的行动方面在特定 Joinpoint
  • 切入点 — — Joinpoint 正则表达式来匹配的。每个时间点加入任何匹配的切入点 , 这个切入点建议相关规定执行
  • 织造 – 的手续方面的针对性建议对象创建一个对象

Spring AOP 和 AspectJ

现在 , 让我们讨论一下 Spring AOP 和 AspectJ 的轴线 — — 诸如能力、目标、编织、内部结构、连接点和质朴。

能力和目标

简单来说 , Spring AOP 和 AspectJ 有不同的目标。

SpringAOP 是 AOP 的简单实现 Spring IoC 来解决该问题 , 最常见的一类。它并不意图作为完整解决方案的 AOP- 它只能应用于被管理的 Bean 时 , Spring 容器。

另一方面 ,AspectJ 是 AOP 技术的原始的 AOP 提供完整解决方案。它更为鲁棒 , 但是也明显多于 Spring AOP 。同时 , 值得注意的是 , 可以应用于所有的 AspectJ 域对象。

织造

Spring AOP 和 AspectJ 都使用了不同类型的编织行为有关 , 严重影响其性能和易用性。

AspectJ 利用三种不同类型的编织 :

  1. 编译时编织AspectJ 编译器 : 输入源代码的应用方面我们和我们的生产和织造类文件作为输出
  2. 织造后编译这也被称为二进制编织。它是用于现有类文件和 JAR 文件 , 我们方面
  3. 装载时织入这正是 : 像原来的二进制编织 , 与编织的差 , 直至类加载器加载到 JVM 中的类文件

更深入的信息对 AspectJ 本身 ,看一下这篇文章

作为使用 AspectJ 编译器 classload 织造时间和时间 ,利用 Spring AOP 的运行时编织

与运行时编织 , 该编织方面在执行应用程序期间使用代理在目标物品 — — JDK 动态代理或使用代理 (在下面讨论的点) :

Comparing Spring AOP and AspectJ

内部结构和应用

Spring AOP 是一个基于代理的 AOP 框架。这意味着 , 为了实现这些方面的目标对象 , 它创建代理对象。这是通过使用以下两种方式 :

  1. JDK 的动态代理 AOP 方式优选为弹簧。即使当目标对象实现一个接口 , 然后使用 JDK 动态代理
  2. CGLIB 代理 – 如果目标对象不实现接口 , 则代理可以使用

我们可以更深入地了解 Spring AOP 代理的机制官方文档

AspectJ 中 , 另一方面 , 起不到任何作用 , 因为类在运行时被编译直接的方面。

Spring AOP 等不同 , 它不需要任何设计图案。在织造方面 , 介绍了其称为编译 AspectJ 编译器 ( ajc ) , 编译和运行程序 , 然后通过小的 (< 100K).

连接点

在 3.3 节中 , 我们显示 , 基于代理的 Spring AOP 模式。正因为如此 , 它需要 Java 类和子类的对象将横切关注点应用。

但它具有局限性。我们不能将横切关注点应用类别 (或方面) 的 “最终” , 因为它们不能被重写 , 从而它将导致运行异常。

这同样适用于静态和最终方法。弹簧方面不能应用 , 因为它们不能被重写。因此 , 因为 Spring AOP 的这些限制 , 只支持方法执行连接点。

然而 ,AspectJ 编织将横切关注点直接转换成实际的代码之前运行。Spring AOP 不同 , 它不需要对象的子类对象 , 并因此支持许多其他的连接点。以下是对支持的连接点 :

Joinpoint Spring AOP 支持 AspectJ 支持
呼叫方法 没有 是的
执行方法 是的 是的
调用构造函数 没有 是的
执行构造函数 没有 是的
执行静态初始化器 没有 是的
对象初始化 没有 是的
参考场 没有 是的
外勤任务 没有 是的
执行处理程序 没有 是的
执行建议 没有 是的

值得注意的是在 spring aop 中 , 问题不是所谓的方法应用于同一类别。

这是因为 , 当我们调用在同一个类中定义的方法 , 则不调用该方法 , Spring AOP 的代理提供。如果我们需要这个功能 , 那么我们必须定义一个不同的咖啡豆中分离方法 , 或者使用 AspectJ 。

简单

Spring AOP 是明显更简单 , 因为它不引入任何额外的编译器或编织器之间的编译过程。它使用运行时编织 , 因此我们通常无缝集成与构建过程。虽然它看上去很简单 , 它只通过弹簧托管 bean 。

然而 , 使用 AspectJ , 我们需要采用 AspectJ 编译器 ( ajc ) 和重新封装库 ( 除非我们所有我们切换到编译后或装入时编织的情况下) 。

当然 , 比前者更复杂 , 因为它介绍了 AspectJ 工具 (包括 Java 编译器 ( ajc ) 、调试器 (ajdb ) 、文件生成器 (ajdoc) 、浏览器程序 ajbrowser 结构 ()) , 我们需要集成或与 IDE 中的构建工具。

性能

就表演而言 ,编译时编织的运行时编织。Spring AOP 是一个基于代理的架构 , 使得代理的存在时启动应用。此外 , 还有几个方面每个方法调用 , 这影响了性能。

另一方面 , AspectJ 中的方面编织到主代码在应用程序执行之前 , 没有额外的运行时开销 , 不像 Spring AOP 。

由于这些原因 ,基准AspectJ 是几乎认为大约 8 到 35 倍 Spring AOP 。

摘要

这个表快速汇总了主要差异, Spring AOP 和 AspectJ 之间进行 :

Spring AOP AspectJ
用纯 Java 使用 Java 编程语言的扩展
不需要单独的编码过程 需要 AspectJ 编译器 ( ajc ) 除非 LTW 设置
唯一可用的运行时编织 织造运行时不可用。支持编译 , 编译后 , 加载时织入
更强大 — — 只支持织制方法 更可编织更强大 — — 字段、方法、构造函数、静态初始化器 , 最终类 / 方法等 …
可以只针对 bean 的 Spring 容器管理 可以实现对所有域对象
仅支持执行切入点方法 所有支持的切入点
代理创建的目标对象 , 同时也应用在这些代理 方面的编织直接转换成代码之前执行应用程序 (在运行时之前)
AspectJ 慢得多 更好的性能
简单易学适用 相对更复杂的是 Spring AOP

选择合适的框架

如果我们在所有分析的论点 , 我们就会明白 , 这根本没有那么一个框架比另一种更好。

简单来说 , 在很大程度上取决于我们的选择要求 :

  • 框架 : 如果不使用 Spring 框架 , 然后我们就别无选择 , 只能放弃这个想法 , 因为它使用的是 Spring AOP 不能管理任何弹簧之外的容器。然而 , 如果我们应用全部创建使用 Spring 框架 , 然后我们可以使用 Spring AOP 的简单的学习和应用
  • 灵活性 : Joinpoint 有限的支持 , 是不完整的 Spring AOP 面向方面编程 (AOP) 的解决方案 , 但它解决了问题 , 最常见的一类。但如果我们想要更深入地挖掘和利用 AOP 以其最大能力的支持 , 希望从广泛可用的连接点 , 则是选择了 AspectJ
  • 表现 : 如果我们使用有限的方面 , 则有轻微的性能差异。但有时会因申请以上几方面。我们不希望使用这种运行时编织的情况下 , 还是选择了 AspectJ 。AspectJ 是 8 至 35 倍 Spring AOP
  • 最好的 : 这些框架的完全兼容。我们总是可以利用 Spring AOP 和 AspectJ 使用时仍可能得到支撑的连接点不支持由前任