写点什么

Spring AOP 内功修炼

作者:4ye
  • 2021 年 11 月 30 日
  • 本文字数:2357 字

    阅读完需:约 8 分钟

今天来和小伙伴们分享这个 Spring AOP 的知识点 👇



这里先推荐大家去阅读 Spring 的官方文档,core 文档的 5,6 章节, 和 AOP 相关的概念都可以这这里找到~,还有我们要了解的 Spring AOP APIs 😄



先来介绍下这个 AOP 😄


AOP 概念

这是个老生常谈的问题呀 哈哈 熟悉的小伙伴们可以略过😄


面向切面编程 Aspect-Oriented Programming ,是对 OOP 的一种补充

使用场景有 : 日志事务权限认证缓存


Spring AOP 呢,会通过这个 代理类 来实现对方法的增强 👇



这里有很重要的几个概念 👇(看图就好啦)



概念关系图


通过 PointCut 去匹配符合的 JoinPoint ,再对其做 Weaving·操作,将 Advice 织入其中。

PointCutAdvice 存放在 Aspect 中。



切点表达式图


如:executionwithinthistargetargsbean

以及匹配注解用的 @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

里面有两个返回类型式 ClassFilterMethodMatcher 的方法,他们的作用如下 👇



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 咱们下期应该……很快再见!! 😆


发布于: 2021 年 11 月 30 日阅读数: 41
用户头像

4ye

关注

公众号:J a v a 4 y e 2021.07.19 加入

定个小目标,写个三年~ 分享一个普通程序员的技术生涯,生活点滴,让学习成为一种习惯!

评论

发布
暂无评论
Spring AOP内功修炼