Spring AOP 内功修炼
今天来和小伙伴们分享这个 Spring AOP 的知识点 👇
这里先推荐大家去阅读 Spring 的官方文档,core 文档的 5,6 章节, 和 AOP 相关的概念都可以这这里找到~,还有我们要了解的 Spring AOP APIs 😄
先来介绍下这个 AOP 😄
AOP 概念
这是个老生常谈的问题呀 哈哈 熟悉的小伙伴们可以略过😄
面向切面编程 Aspect-Oriented Programming ,是对 OOP 的一种补充
使用场景有 : 日志 ,事务 , 权限认证 , 缓存 等
而 Spring AOP 呢,会通过这个 代理类 来实现对方法的增强 👇
这里有很重要的几个概念 👇(看图就好啦)
概念关系图
通过 PointCut 去匹配符合的 JoinPoint ,再对其做 Weaving·操作,将 Advice 织入其中。
而 PointCut 和 Advice 存放在 Aspect 中。
切点表达式图
如:execution ,within , this ,target ,args ,bean
以及匹配注解用的 @target ,@args ,@within , @annotation
execution
作用: 匹配 某些连接点
如:
execution(public * *(..)) 作用: 匹配所有公共方法
execution(* set*(..)) 作用:匹配所有 set 开头的方法
execution(* com.xyz.service.AccountService.*(..)) 作用:匹配 AccountService 中的所有方法
execution(* com.xyz.service..(..)) 作用:匹配 service 包中的所有方法
execution(* com.xyz.service...(..)) 作用:匹配 service 包和它的子包中 所有的方法
within
作用:匹配 类 中的所有 连接点
如:
within(com.xyz.service.*) 作用:匹配 service 包中的所有方法
within(com.xyz.service..*) 作用:匹配 service 包和它的子包中 所有的方法
this
作用:匹配 代理类
this(com.xyz.service.AccountService) 作用:匹配 实现了 AccountService 接口的 代理类,即 父类是 AccountService
target
作用:匹配 被代理类
target(com.xyz.service.AccountService) 作用:匹配 实现了 AccountService 接口的 被代理类,即 父类是 AccountService
args
作用:匹配 方法参数
args(java.io.Serializable) 作用:匹配 单个方法参数,且是 Serializable 类型的
bean
作用:匹配 bean 名称,Spring 扩展 AspectJ 的 PDC 规范, 只能在 Spring AOP 中生效
@target
作用:匹配 有注解的 被代理类
@target(org.springframework.transaction.annotation.Transactional) 作用:匹配 @Transactional 注解的 被代理类
@args
作用:匹配 传入的参数 有某个注解
@args(com.xyz.security.Classified) 作用: 匹配有 @Classified 注解的参数
@within
作用:匹配 有某个注解 的 类
@within(org.springframework.transaction.annotation.Transactional) 作用: 匹配 有 @Transactional 注解 的 类中的所有 连接点
@annotation
作用:匹配 该注解
@annotation(org.springframework.transaction.annotation.Transactional) 作用: 匹配 @Transactional 注解
小结图如下 👇
Advice 注解图
这里是常用的五个注解
注意:@After 等于 finally 语句块,必定会执行,还有 @Around 注解时,方法的第一个参数必须是 ProceedingJoinPoint ,它是 JoinPoint 的子接口
下面的内容是本文的重点~ 😝
编程式 AOP
这个呢,是我们写 AOP 插件 的基础
这里要理解这几个概念 👇
Pointcut : 对应 @Pointcut
Advice :对应 @Before,@After,@Around 等等
Advisor :@Aspect 注解解析后的类,等于 Pointcut + Advice 🐷
Advised : Spring AOP 代理类都可以转换成该接口! 🐷
例子我放到 Github 上啦 ,地址放文末啦👇
Pointcut
里面有两个返回类型式 ClassFilter 和 MethodMatcher 的方法,他们的作用如下 👇
4ye 也是大概了解下 哈哈 主要还是用 Spring 提供的实现类,如下面例子中用到的 NameMatchMethodPointcut ,看名字就能猜个大概,就是匹配这个方法名的 😄
小例子
代码例子比较多,这里就举这个 Pointcut 看看就好啦~ (其他案例代码小伙伴们可以自行在我的 Github 仓库上获取😄)
通过对 ProxyFactroy 添加 切面 来对目标 sayService 做增强,这里的 Pointcut 只针对 setName 方法
效果
可以看到只增强了 setName 方法
Advice
可以看到这里 Spring 提供了这 五种类型。除了最后一种 Introduction 类型上面没提到外,是不是还发现少了一种类型😝
没错,就是这个 @After 注解对应的最终通知 😱
这去哪了呀 ( ̄﹏ ̄;)
在源码中寻找一番后,发现了这个 AspectJAfterAdvice 👇
可以看到这里不仅实现了 AfterAdvice 接口,还要实现这个 MethodInterceptor 接口 ,最后真的是在 finally 语句块中实现了这个增强操作,来达到 最终通知 的效果!
所以我们需要 最终通知 效果的话,可以直接模仿它去实现就好了~
而 Introduction Advice 这个还挺特别的🐖
作用:为代理类实现新的接口
等于 @DeclareParents 注解
Advisor
作用 👇
@Aspect 注解解析后的类,等于 Pointcut + Advice
Advised
作用 👇
AOP 代理类都可以转换成该接口
为啥这些代理对象都能转换成这个 Advised 接口?
答案当然不言而喻 哈哈,咱们来源码中看看 Spring 是怎么写的😝
JDK
Cglib
结论
可以发现,不管是 JDK 还是 Cglib , 都会去调用同一个方法,为代理类多实现几个接口,其中一个就是 Advised 😋
关系图
最后
本文就分享到这里啦🐖
重点内容在 编程式 AOP 这一块~ 内功+1 😋
代码可以在这里获取 👇 可以自己多跑跑代码,会有更多问题等着你发现的!
https://github.com/Java4ye/springboot-demo-4ye/tree/main/springboot-aop
思维导图 👇
http://processon.com/chart_image/6134d6cb7d9c081c753b5d54.png
ps:这次分享的时候居然出现了 Gateway Timeout 异常 ,如果等不及官方修复的话,可以在网盘获取
链接:https://pan.baidu.com/s/1KVWw0DaYCKrISIKJgqqyuA
提取码 需在公众号后台回复 bdwp
插件的部分代码我也上传啦,就差这画画图写写文啦 哈哈 (这两天应该就安排上了)😆
喜欢的话可以 关注 并 星标 下公众号 Java4ye 支持下 4ye 呀😝,这样就可以第一时间收到更文消息啦🐷
我是 4ye 咱们下期应该……很快再见!! 😆
版权声明: 本文为 InfoQ 作者【4ye】的原创文章。
原文链接:【http://xie.infoq.cn/article/c110a8e97af3a21746b1bbacd】。
本文遵守【CC-BY 4.0】协议,转载请保留原文出处及本版权声明。
评论