我工作遇到的第一个问题就是用 Spring AOP 解决的。Spring AOP 真正用起来的时候很少,但是无一不是解决重要问题。用一种轻松的手法解决了重要的问题,Spring AOP 做到了。
Spring AOP 的登场
在工作中经常会遇到维护老 WEB 项目的任务,老项目有很多问题,其中一个就是没有日志系统,出现问题,排查困难。当时我的任务就是给老项目增加日志。按理来说,增加日志不难,增加 Log4j 依赖,填好配置。在每个 Controller 类和 Server 类里创建日志对象,然后在关键位置上加上日志打印就行。
问题在于,项目已经写好了,总共大好几千的方法一个个加日志打印,人要累吐血,而且这属于重复工作,太痛苦了。于是就想到了使用 Spring AOP 添加日志,果然轻易的解决了问题。
什么是 Spring AOP
现在进入正题,首先介绍什么是 Spring AOP。官方介绍是面向切面编程。首先这是一种编程思想,基于面向对象编程(OOP)的一种补充。面向对象编程思想是指万物皆对象。处理对象的过程叫方法,面向切面编程就是指把方法看成一个整体,在不改变方法原代码的情况下改变方法执行过程。
打个比方就是番茄夹梅子。先把番茄切开,里面放一个梅子,然后把番茄合上,番茄还是那个番茄,没有变,但是里面有了梅子的味道。在物理层面,番茄已经改变了,里面多了个梅子。但是变成代码的时候,我们可以在不改变方法源代码的情况下,改变方法运行逻辑。
Spring AOP 原理
这种神奇的效果是怎么实现的呢?这个要归功于 23 种设计模式中的代理模式。要理解面向切面编程的原理,必须要先理解代理模式。
代理模式理解起来很简单,打个比方就明白了。比如说你是彭于晏(读者人均彭于晏),但是你只专注于表演这个能力,只能拿到演戏的钱。现在你特别出名,好多公司找你做代言人。但是你没空去洽谈。这个时候你就需要一个经纪人,经纪人会帮你接代言,帮你管理粉丝,帮你接戏。你的工作还是没变,但是收入增加了。
在这个过程中,“彭于晏”代表的是原生对象。而全权代表“彭于晏”的经纪人被称为代理对象。当我需要“彭于晏”表演的时候,我去找经纪人。这个过程就是代理模式。
流程图:
代理模式的两种常见实现方法:动态代理、静态代理
先说静态代理:静态代理的优点是,好理解,好观察。缺点是难维护,死板。不适合在动态的项目需求中使用。
简易版静态代理:
这就是静态代理,目标对象和代理对象都需要实现自己的父类方法。一旦父类增加或修改了,需要在所有子类中一起变动,维护十分困难。所以一般在项目中很少使用静态代理。
再说动态代理:动态代理的优点是:好使用,好维护。缺点是:不好理解,不好观测。现在上代码:
简易版动态代理:接口部分不变,“彭于晏”部分不变
注意接下来的代理对象
所谓动态代理就是借用反射来创建代理对象,借用反射来创建代理对象的好处就是通用性极高。这样就避免了上层接口改变,代理对象也要修改代码这一蛋疼行为。
Spring AOP 上手使用
现在明白了 Spring AOP 是干啥的,接下来展示,Spring AOP 在实际项目中的使用。Spring AOP 已经实现了自动创建动态代理,我们只需要学会正确调用即可。
图文展示:
第一步:导入 Spring AOP 依赖
PS:加入新依赖后要更新依赖
第二步:使用切面注解@Aspect
切面:定义当前类是切面类,需要给每一个满足切面条件的类生成不同的代理对象。
第三步:定义需要的通知。
通知:通知就是定义代理对象的代码要在什么时候生效。
以下是常用通知类型,可以根据实际需要选用通知
第四步:填写需要的切面表达式
切面表达式:定义了满足什么条件就去生成其代理类。
整理常用切点表达式:
执行结果:
Spring AOP 的优势和缺点
Spring AOP 的优势很明显,在不改变原代码的情况下,加入新的逻辑代码。缺点是,逻辑不连贯,阅读体验不好。毕竟这是专门为解决特定问题而出现的专用工具。
我们应该有一个意识,当我们需要批量给方法新增逻辑的时候,要能想起 Spring AOP。
Java 里面概念很多,知识点也很多。很多概念和知识点是专为解决特定问题而诞生的,我们不可能记住全部知识点,但是在需要用到一个知识点的时候,要能想起什么知识点能解决这个问题,在不断地遇到问题,解决问题的过程中,我们对这些知识点的理解自然会更深刻,但是不要停止脚步,要多想原理和为什么,这是通往架构师的道路。
Spring AOP 总结
回顾一下本章重点:
AOP 是一种编程思想——面向切面编程
AOP 实现原理——代理模式。代理模式分为动态代理和静态代理。
Spring AOP 使用步骤:1.导入依赖。2.使用切面注释。3.使用通知注释。4.使用切点申明表达式。
都看到这里了。英俊潇洒的小哥哥,貌美如花小姐姐,看完记得顺手点赞。